-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathquicksolve.c
252 lines (205 loc) · 8.25 KB
/
quicksolve.c
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
#include <stdio.h>
#include <unistd.h>
#include <stdbool.h>
#include <stdlib.h>
#include <signal.h>
#include <string.h>
#include "src/integral.h"
#include "src/integralmgr.h"
#include "src/operand.h"
#include "src/coefficient.h"
#include "src/print.h"
#include "src/pivotgraph.h"
#define DEF_NUM_PROCESSORS 1
#define DEF_PREALLOC 1<<20
#define DEF_MEMLIMIT 0
#define DEF_BACKING "storage.dat#type=kch"
#define DEF_FERCYCLE 0
#define DEF_LIMITTERMINALS 0
#define STR( X ) #X
#define XSTR( X ) STR( X )
const char const usage[ ]= "[-p <Symbolic threads>] [-n <Numeric threads>] [-k <Fermat cycle>] [-a <Identity limit>] [-m <Memory limit>] [-e <Elimination Mode>] [-t <Terminal Limit>] [-b <Backing DB>] [-q] [<Symbol><Assignment><Substitution>] ...]\n\n"
"<Symbolic threads>: Number of evaluators in parallel calculation of symbolic expressions [Default " XSTR( DEF_NUM_PROCESSORS ) "]\n"
"<Numeric threads>: Number of evaluators in parallel calculation of numeric expressions [Default " XSTR( DEF_NUM_PROCESSORS ) "]\n"
"<Identity limit>: Upper bound on the number of identities in the system [Default " XSTR( DEF_PREALLOC ) "]\n"
"<Memory limit>: Memory limit in bytes above which coefficients are written to disk backing space or 0 for no limit [Default " XSTR( DEF_MEMLIMIT )"]\n"
"<Elimination Mode>: How to deal with numerical zeroes. Either of: discard them (o)ptimistacally, (w)ait for the symbolic result, or (k)eep them [Default k]\n"
"<Terminal limit>: Limit of number of evaluated coefficients to accumulate in the symbolic tree of operations [Default " XSTR( DEF_LIMITTERMINALS )"]\n"
"<Backing DB>: Kyotocabinet formatted string indicating the disk backing space database [Default '" DEF_BACKING "']\n"
"<Symbol>: One of the symbols occurring in the databases. All symbols must be registered\n"
"<Assignment>: Either '=' for numeric assignment only or ':' to substitute the given value even in the symbolic result\n"
"<Substitution>: The value to be substituted for the associated symbol\n"
"<Fermat cycle>: If greater than 0, reinitializes the FERMAT backend every that many evaluations [Default " XSTR( DEF_FERCYCLE ) "]\n\n"
"Reads list of integrals from stdin and produces FORM fill statements for those integrals in terms of master integrals to stdout. All occurring symbols from the identity databases must be registered as positional arguments and can optionally be chosen to be replaced.\n"
"If -q is given, Quicksolve will not wait for finalization of each solution to print them but will only report that a solution has been formally obtained and may possibly still be evaluating.\n\n"
"For further documentation see the manual that came with Quicksolve";
#include "src/policies/cks.c"
struct CKSInfo info = { NULL,false,ELIMINATE_NONE };
void signalled( int signum ) {
info.terminate = true;
}
int main( const int argc,char* const argv[ ] ) {
// Parse arguments
int num_processors = DEF_NUM_PROCESSORS;
int num_processors_numeric = DEF_NUM_PROCESSORS;
unsigned prealloc = DEF_PREALLOC;
size_t memlimit = DEF_MEMLIMIT;
size_t limit_terminals = DEF_LIMITTERMINALS;
char* storage = DEF_BACKING;
unsigned fercycle = DEF_FERCYCLE;
bool quiet = false;
bool help = false;
FILE* const infile = stdin;
FILE* const outfile = stdout;
int opt;
while( ( opt = getopt( argc,argv,"p:a:hqk:b:m:t:e:n:" ) )!=-1 ) {
char* endptr;
switch( opt ) {
case 'p':
if( ( num_processors = strtol( optarg,&endptr,0 ) )<1 || *endptr!='\0' )
help = true;
break;
case 'n':
if( ( num_processors_numeric = strtol( optarg,&endptr,0 ) )<1 || *endptr!='\0' )
help = true;
break;
case 'a':
if( ( prealloc = strtol( optarg,&endptr,0 ) )<0 || *endptr!='\0' )
help = true;
break;
case 'k':
if( ( fercycle = strtol( optarg,&endptr,0 ) )<0 || *endptr!='\0' )
help = true;
break;
case 'b':
storage = optarg;
break;
case 'm':
if( ( memlimit = strtol( optarg,&endptr,0 ) )<0 || *endptr!='\0' )
help = true;
break;
case 't':
if( ( limit_terminals = strtol( optarg,&endptr,0 ) )<0 || *endptr!='\0' )
help = true;
break;
case 'e':
if( optarg[ 0 ]=='o' )
info.elimination = ELIMINATE_OPTIMISTIC;
else if( optarg[ 0 ]=='w' )
info.elimination = ELIMINATE_WAIT;
break;
case 'h':
help = true;
break;
case 'q':
quiet = true;
break;
}
}
QsDb storage_db = qs_db_new( storage,QS_DB_WRITE | QS_DB_CREATE );
if( help || !storage_db ) {
printf( "%s %s\n",argv[ 0 ],usage );
exit( EXIT_FAILURE );
}
// Reap fermat processes immediately
sigaction( SIGCHLD,&(struct sigaction){ .sa_handler = SIG_IGN,.sa_flags = SA_NOCLDWAIT },NULL );
// Ignore broken pipe to Fermat, we detect this by premature EOF
sigaction( SIGPIPE,&(struct sigaction){ .sa_handler = SIG_IGN },NULL );
// Handle request for termination
sigaction( SIGINT,&(struct sigaction){ .sa_handler = signalled },NULL );
// Disable buffering for accurate timing
setbuf( stdout,NULL );
QsEvaluatorOptions fermat_options = qs_evaluator_options_new( );
QsEvaluatorOptions fermat_options_numeric = qs_evaluator_options_new( );
QsIntegralMgr mgr = qs_integral_mgr_new_with_size( "idPR",".dat#type=kch","PR",".dat#type=kch",prealloc );
int j;
for( j = optind; j<argc; j++ ) {
char* separator_equ = strchr( argv[ j ],'=' );
char* separator_col = strchr( argv[ j ],':' );
assert( separator_equ || separator_col );
char* symbol = argv[ j ];
char* value;
char* value_numeric;
if( !separator_equ ||( separator_col && separator_col<separator_equ ) ) {
/* Full replacement */
*separator_col = '\0';
value_numeric = separator_col + 1;
value = value;
} else {
/* Only numeric replacement */
*separator_equ = '\0';
value_numeric = separator_equ + 1;
value = NULL;
}
qs_evaluator_options_add( fermat_options,symbol,value );
qs_evaluator_options_add( fermat_options_numeric,symbol,value_numeric );
}
qs_evaluator_options_add( fermat_options,"#",fercycle );
qs_evaluator_options_add( fermat_options,"!",FERMAT_BINARY );
qs_evaluator_options_add( fermat_options_numeric,"!",FERMAT_BINARY_NUMERIC );
#if QS_STATUS
QsAEF aef = qs_aef_new( limit_terminals,false );
QsAEF aef_numeric = qs_aef_new( 0,true );
#else
QsAEF aef = qs_aef_new( limit_terminals );
QsAEF aef_numeric = qs_aef_new( 0 );
#endif
info.graph = qs_pivot_graph_new_with_size( aef,aef_numeric,mgr,(QsLoadFunction)qs_integral_mgr_load_expression,mgr,(QsSaveFunction)qs_integral_mgr_save_expression,storage_db,memlimit,prealloc );
for( j = 0; j<num_processors; j++ )
qs_aef_spawn( aef,fermat_options );
for( j = 0; j<num_processors_numeric; j++ )
qs_aef_spawn( aef_numeric,fermat_options_numeric );
qs_evaluator_options_destroy( fermat_options );
qs_evaluator_options_destroy( fermat_options_numeric );
ssize_t chars;
size_t N = 0;
char* buffer = NULL;
while( ( chars = getline( &buffer,&N,infile ) )!=-1 ) {
QsIntegral i = qs_integral_new_from_string( buffer );
if( i ) {
QsComponent id = qs_integral_mgr_manage( mgr,i );
cks_solve( &info,id );
if( info.terminate )
break;
char* target;
qs_integral_print( qs_integral_mgr_peek( mgr,id ),&target );
if( quiet )
fprintf( outfile,"%s\n",target );
else {
struct QsReflist result = qs_pivot_graph_acquire( info.graph,id );
if( result.references ) {
fprintf( outfile,"fill %s =",target );
if( result.n_references>1 ) {
int j;
for( j = 0; j<result.n_references; j++ )
if( result.references[ j ].head!=id ) {
char* head,* coeff;
qs_integral_print( qs_integral_mgr_peek( mgr,result.references[ j ].head ),&head );
qs_coefficient_print( result.references[ j ].coefficient,&coeff );
fprintf( outfile,"\n + %s * (%s)",head,coeff );
free( coeff );
free( head );
}
} else
fprintf( outfile,"\n0" );
fprintf( outfile,"\n;\n" );
fflush( outfile );
free( result.references );
qs_pivot_graph_release( info.graph,id );
}
}
free( target );
} else
fprintf( stderr,"Warning: Could not parse '%s'\n",buffer );
}
free( buffer );
DBG_PRINT( "Solution done. Finalizing\n",0 );
qs_pivot_graph_destroy( info.graph );
qs_aef_destroy( aef );
qs_aef_destroy( aef_numeric );
qs_integral_mgr_destroy( mgr );
qs_db_destroy( storage_db );
fclose( infile );
fclose( outfile );
exit( EXIT_SUCCESS );
}