source: git/Singular/ssiLink.cc @ 3857cb

fieker-DuValspielwiese
Last change on this file since 3857cb was 3857cb, checked in by Hans Schoenemann <hannes@…>, 13 years ago
waitpid fix for ssi git-svn-id: file:///usr/local/Singular/svn/trunk@14224 2c84dea3-7e68-4137-9b89-c4e89433aadc
  • Property mode set to 100644
File size: 39.9 KB
Line 
1/****************************************
2 * Computer Algebra System SINGULAR     *
3 ****************************************/
4/***************************************************************
5 * File:    ssiLink.h
6 *  Purpose: declaration of sl_link routines for ssi
7 *  Version: $Id$
8 ***************************************************************/
9#include <stdio.h>
10#include <fcntl.h>
11#include <errno.h>
12#include <unistd.h>
13#include <sys/types.h>
14#include <signal.h>
15#include <sys/types.h>          /* for portability */
16#include <sys/select.h>
17#include <sys/socket.h>
18#include <netinet/in.h>
19#include <ctype.h>   /*for isdigit*/
20#include <netdb.h>
21#include <sys/wait.h>
22
23
24
25#include <kernel/mod2.h>
26// #include "mod2.h"
27
28#include <Singular/tok.h>
29#include <Singular/ipid.h>
30#include <omalloc/omalloc.h>
31#include <kernel/ring.h>
32#include <kernel/matpol.h>
33#include <kernel/ideals.h>
34#include <kernel/polys.h>
35#include <kernel/longrat.h>
36#include <kernel/ideals.h>
37#include <kernel/intvec.h>
38#include <kernel/options.h>
39#include <kernel/timer.h>
40#include <Singular/subexpr.h>
41#include <Singular/silink.h>
42#include <Singular/cntrlc.h>
43#include <Singular/lists.h>
44#include <Singular/blackbox.h>
45#include <Singular/ssiLink.h>
46
47#ifdef HAVE_MPSR
48#include <Singular/mpsr.h>
49#endif
50
51//#if (_POSIX_C_SOURCE >= 200112L) || (_XOPEN_SOURCE >= 600)
52//#define HAVE_PSELECT
53//#endif
54
55#define SSI_VERSION 2
56
57typedef struct
58{
59  FILE *f_read;
60  FILE *f_write;
61  ring r;
62  pid_t pid; /* only valid for fork/tcp mode*/
63  int fd_read,fd_write; /* only valid for fork/tcp mode*/
64  char level;
65  char ungetc_buf; /* status sets to !=0, if ungetc was used, ssiRead* set to 0*/
66  char send_quit_at_exit;
67
68} ssiInfo;
69
70
71link_list ssiToBeClosed=NULL;
72sigset_t ssi_sigmask;
73sigset_t ssi_oldmask;
74#define SSI_BLOCK_CHLD sigprocmask(SIG_SETMASK, &ssi_sigmask, &ssi_oldmask)
75#define SSI_UNBLOCK_CHLD sigprocmask(SIG_SETMASK, &ssi_oldmask, NULL)
76
77// the helper functions:
78void ssiSetCurrRing(const ring r)
79{
80  if (!rEqual(r,currRing,1))
81  {
82    char name[20];
83    int nr=0;
84    do
85    { sprintf(name,"ssiRing%d",nr); nr++; }
86    while(IDROOT->get(name, 0)!=NULL);
87    idhdl h=enterid(omStrDup(name),0,RING_CMD,&IDROOT,FALSE);
88    IDRING(h)=r;
89    r->ref++;
90    rSetHdl(h);
91  }
92}
93// the implementation of the functions:
94void ssiWriteInt(ssiInfo *d,const int i)
95{
96  SSI_BLOCK_CHLD;
97  fprintf(d->f_write,"%d ",i);
98  SSI_UNBLOCK_CHLD;
99  //if (d->f_debug!=NULL) fprintf(d->f_debug,"int: %d ",i);
100}
101
102void ssiWriteString(ssiInfo *d,const char *s)
103{
104  SSI_BLOCK_CHLD;
105  fprintf(d->f_write,"%d %s ",strlen(s),s);
106  SSI_UNBLOCK_CHLD;
107  //if (d->f_debug!=NULL) fprintf(d->f_debug,"stringi: %d \"%s\" ",strlen(s),s);
108}
109
110
111void ssiWriteBigInt(const ssiInfo *d, const number n)
112{
113  // syntax is as follows:
114  // case 2 Q:     3 4 <int>
115  //        or     3 3 <mpz_t nominator>
116  SSI_BLOCK_CHLD;
117  if(SR_HDL(n) & SR_INT)
118  {
119    fprintf(d->f_write,"4 %ld ",SR_TO_INT(n));
120    //if (d->f_debug!=NULL) fprintf(d->f_debug,"bigint: short \"%ld\" ",SR_TO_INT(n));
121  }
122  else if (n->s==3)
123  {
124    fprintf(d->f_write,"3 ");
125    mpz_out_str(d->f_write,10,n->z);
126    fprintf(d->f_write," ");
127    //gmp_fprintf(d->f_write,"3 %Zd ",n->z);
128    //if (d->f_debug!=NULL) gmp_fprintf(d->f_debug,"bigint: gmp \"%Zd\" ",n->z);
129  }
130  else Werror("illiegal bigint");
131  SSI_UNBLOCK_CHLD;
132}
133
134void ssiWriteNumber(const ssiInfo *d, const number n)
135{
136  // syntax is as follows:
137  // case 1 Z/p:   3 <int>
138  // case 2 Q:     3 4 <int>
139  //        or     3 0 <mpz_t nominator> <mpz_t denominator>
140  //        or     3 1  dto.
141  //        or     3 3 <mpz_t nominator>
142  //        or     3 5 <mpz_t raw nom.> <mpz_t raw denom.>
143  //        or     3 6 <mpz_t raw nom.> <mpz_t raw denom.>
144  //        or     3 7 <mpz_t raw nom.>
145  SSI_BLOCK_CHLD;
146  if(rField_is_Zp(d->r))
147  {
148    fprintf(d->f_write,"%d ",(int)(long)n);
149    //if (d->f_debug!=NULL) fprintf(d->f_debug,"number: \"%ld\" ",(int)(long)n);
150  }
151  else if (rField_is_Q(d->r))
152  {
153    if(SR_HDL(n) & SR_INT)
154    {
155      fprintf(d->f_write,"4 %ld ",SR_TO_INT(n));
156      //if (d->f_debug!=NULL) fprintf(d->f_debug,"number: short \"%ld\" ",SR_TO_INT(n));
157    }
158    else if (n->s<2)
159    {
160      //gmp_fprintf(d->f_write,"%d %Zd %Zd ",n->s,n->z,n->n);
161      fprintf(d->f_write,"%d ",n->s+5);
162      mpz_out_str (d->f_write,32, n->z);
163      fprintf(d->f_write," ");
164      mpz_out_str (d->f_write,32, n->n);
165      fprintf(d->f_write," ");
166
167      //if (d->f_debug!=NULL) gmp_fprintf(d->f_debug,"number: s=%d gmp/gmp \"%Zd %Zd\" ",n->s,n->z,n->n);
168    }
169    else /*n->s==3*/
170    {
171      //gmp_fprintf(d->f_write,"3 %Zd ",n->z);
172      fprintf(d->f_write,"8 ");
173      mpz_out_str (d->f_write,32, n->z);
174      fprintf(d->f_write," ");
175
176      //if (d->f_debug!=NULL) gmp_fprintf(d->f_debug,"number: gmp \"%Zd\" ",n->z);
177    }
178  }
179  else WerrorS("coeff field not implemented");
180  SSI_UNBLOCK_CHLD;
181}
182
183void ssiWriteRing(ssiInfo *d,const ring r)
184{
185  /* 5 <ch> <N> <l1> <v1> ...<lN> <vN> <number of orderings> <ord1> <block0_1> <block1_1> .... */
186  if (d->r!=NULL) rKill(d->r);
187  d->r=r;
188  d->r->ref++;
189  SSI_BLOCK_CHLD;
190  fprintf(d->f_write,"%d %d ",r->ch,r->N);
191
192  int i;
193  for(i=0;i<r->N;i++)
194  {
195    fprintf(d->f_write,"%d %s ",strlen(r->names[i]),r->names[i]);
196  }
197  /* number of orderings:*/
198  i=0;
199  while (r->order[i]!=0) i++;
200  fprintf(d->f_write,"%d ",i);
201  /* each ordering block: */
202  i=0;
203  while(r->order[i]!=0)
204  {
205    fprintf(d->f_write,"%d %d %d ",r->order[i],r->block0[i], r->block1[i]);
206    i++;
207  }
208  SSI_UNBLOCK_CHLD;
209}
210
211void ssiWritePoly(ssiInfo *d, int typ, poly p)
212{
213  SSI_BLOCK_CHLD;
214  fprintf(d->f_write,"%d ",pLength(p));//number of terms
215  SSI_UNBLOCK_CHLD;
216  int i;
217
218  while(p!=NULL)
219  {
220    ssiWriteNumber(d,pGetCoeff(p));
221    //nWrite(fich,pGetCoeff(p));
222    SSI_BLOCK_CHLD;
223    fprintf(d->f_write,"%ld ",p_GetComp(p,d->r));//component
224
225    for(int j=1;j<=rVar(d->r);j++)
226    {
227      fprintf(d->f_write,"%ld ",p_GetExp(p,j,d->r ));//x^j
228    }
229    pIter(p);
230    SSI_UNBLOCK_CHLD;
231  }
232}
233
234void ssiWriteIdeal(ssiInfo *d, int typ,ideal I)
235{
236   // syntax: 7 # of elements <poly 1> <poly2>.....
237   // syntax: 8 <rows> <cols> <poly 1> <poly2>.....
238   matrix M=(matrix)I;
239   SSI_BLOCK_CHLD;
240   if (typ==MATRIX_CMD)
241        fprintf(d->f_write,"%d %d ", MATROWS(M),MATCOLS(M));
242   else
243     fprintf(d->f_write,"%d ",IDELEMS(I));
244    SSI_UNBLOCK_CHLD;
245
246   int i;
247   int tt;
248   if (typ==MODUL_CMD) tt=VECTOR_CMD;
249   else                tt=POLY_CMD;
250
251   for(i=0;i<IDELEMS(I);i++)
252   {
253     ssiWritePoly(d,tt,I->m[i]);
254   }
255}
256void ssiWriteCommand(si_link l, command D)
257{
258  ssiInfo *d=(ssiInfo*)l->data;
259  // syntax: <num ops> <operation> <op1> <op2> ....
260  SSI_BLOCK_CHLD;
261  fprintf(d->f_write,"%d %d ",D->argc,D->op);
262  SSI_UNBLOCK_CHLD;
263  if (D->argc >0) ssiWrite(l, &(D->arg1));
264  if (D->argc < 4)
265  {
266    if (D->argc >1) ssiWrite(l, &(D->arg2));
267    if (D->argc >2) ssiWrite(l, &(D->arg3));
268  }
269}
270
271void ssiWriteProc(ssiInfo *d,procinfov p)
272{
273  ssiWriteString(d,p->data.s.body);
274}
275
276void ssiWriteList(si_link l,lists dd)
277{
278  ssiInfo *d=(ssiInfo*)l->data;
279  SSI_BLOCK_CHLD;
280  fprintf(d->f_write,"%d ",dd->nr+1);
281  SSI_UNBLOCK_CHLD;
282  int i;
283  for(i=0;i<=dd->nr;i++)
284  {
285    ssiWrite(l,&(dd->m[i]));
286  }
287}
288void ssiWriteIntvec(ssiInfo *d,intvec * v)
289{
290  SSI_BLOCK_CHLD;
291  fprintf(d->f_write,"%d ",v->length());
292  int i;
293  for(i=0;i<v->length();i++)
294  {
295    fprintf(d->f_write,"%d ",(*v)[i]);
296  }
297  SSI_UNBLOCK_CHLD;
298}
299
300char *ssiReadString(ssiInfo *d)
301{
302  char *buf;
303  int l;
304  SSI_BLOCK_CHLD;
305  fscanf(d->f_read,"%d ",&l);
306  SSI_UNBLOCK_CHLD;
307  buf=(char*)omAlloc(l+1);
308  SSI_BLOCK_CHLD;
309  fread(buf,1,l,d->f_read);
310  SSI_UNBLOCK_CHLD;
311  buf[l]='\0';
312  return buf;
313}
314
315int ssiReadInt(FILE *fich)
316{
317  int d;
318  SSI_BLOCK_CHLD;
319  fscanf(fich,"%d",&d);
320  SSI_UNBLOCK_CHLD;
321  return d;
322}
323
324number ssiReadBigInt(ssiInfo *d)
325{
326  int sub_type=-1;
327  SSI_BLOCK_CHLD;
328  fscanf(d->f_read,"%d",&sub_type);
329  SSI_UNBLOCK_CHLD;
330  switch(sub_type)
331  {
332   case 3:
333     {// read int or mpz_t or mpz_t, mpz_t
334       number n=nlRInit(0);
335       SSI_BLOCK_CHLD;
336       mpz_inp_str(n->z,d->f_read,0);
337       SSI_UNBLOCK_CHLD;
338       n->s=sub_type;
339       return n;
340     }
341   case 4:
342     {
343       int dd;
344       SSI_BLOCK_CHLD;
345       fscanf(d->f_read,"%d",&dd);
346       SSI_UNBLOCK_CHLD;
347       return INT_TO_SR(dd);
348     }
349   default:
350       Werror("error in reading bigint: invalid subtype %d",sub_type);
351       return NULL;
352   }
353}
354
355number ssiReadNumber(ssiInfo *d)
356{
357  if (rField_is_Q(d->r))
358  {
359     int sub_type=-1;
360     SSI_BLOCK_CHLD;
361     fscanf(d->f_read,"%d",&sub_type);
362     SSI_UNBLOCK_CHLD;
363     switch(sub_type)
364     {
365     case 0:
366     case 1:
367       {// read mpz_t, mpz_t
368         number n=nlRInit(0);
369         mpz_init(n->n);
370         SSI_BLOCK_CHLD;
371         gmp_fscanf(d->f_read,"%Zd %Zd",n->z,n->n);
372         SSI_UNBLOCK_CHLD;
373         n->s=sub_type;
374         return n;
375       }
376
377     case 3:
378       {// read mpz_t
379         number n=nlRInit(0);
380         SSI_BLOCK_CHLD;
381         gmp_fscanf(d->f_read,"%Zd",n->z);
382         SSI_UNBLOCK_CHLD;
383         n->s=3; /*sub_type*/
384         return n;
385       }
386     case 4:
387       {
388         int dd;
389         SSI_BLOCK_CHLD;
390         fscanf(d->f_read,"%d",&dd);
391         SSI_UNBLOCK_CHLD;
392         return INT_TO_SR(dd);
393       }
394     case 5:
395     case 6:
396       {// read raw mpz_t, mpz_t
397         number n=nlRInit(0);
398         mpz_init(n->n);
399         SSI_BLOCK_CHLD;
400         mpz_inp_str (n->z, d->f_read, 32);
401         mpz_inp_str (n->n, d->f_read, 32);
402         SSI_UNBLOCK_CHLD;
403         n->s=sub_type-5;
404         return n;
405       }
406     case 8:
407       {// read raw mpz_t
408         number n=nlRInit(0);
409         SSI_BLOCK_CHLD;
410         mpz_inp_str (n->z, d->f_read, 32);
411         SSI_UNBLOCK_CHLD;
412         n->s=sub_type=3; /*subtype-5*/
413         return n;
414       }
415
416     default: Werror("error in reading number: invalid subtype %d",sub_type);
417              return NULL;
418     }
419  }
420  else if (rField_is_Zp(d->r))
421  {
422    // read int
423    int dd;
424    SSI_BLOCK_CHLD;
425    fscanf(d->f_read,"%d",&dd);
426    SSI_UNBLOCK_CHLD;
427    return (number)dd;
428  }
429  else Werror("coeffs not implemented");
430  return NULL;
431}
432
433ring ssiReadRing(ssiInfo *d)
434{
435/* syntax is <ch> <N> <l1> <v1> ...<lN> <vN> <number of orderings> <ord1> <block0_1> <block1_1> .... */
436  int ch, N,i,l;
437  char **names;
438  SSI_BLOCK_CHLD;
439  fscanf(d->f_read,"%d %d ",&ch,&N);
440  SSI_UNBLOCK_CHLD;
441  names=(char**)omAlloc(N*sizeof(char*));
442  for(i=0;i<N;i++)
443  {
444    names[i]=ssiReadString(d);
445  }
446  // read the orderings:
447  int num_ord; // number of orderings
448  SSI_BLOCK_CHLD;
449  fscanf(d->f_read,"%d",&num_ord);
450  SSI_UNBLOCK_CHLD;
451  int *ord=(int *)omAlloc0((num_ord+1)*sizeof(int));
452  int *block0=(int *)omAlloc0((num_ord+1)*sizeof(int));
453  int *block1=(int *)omAlloc0((num_ord+1)*sizeof(int));
454  SSI_BLOCK_CHLD;
455  for(i=0;i<num_ord;i++)
456  {
457     fscanf(d->f_read,"%d %d %d",&ord[i],&block0[i],&block1[i]);
458  }
459  SSI_UNBLOCK_CHLD;
460  return rDefault(ch,N,names,num_ord,ord,block0,block1);
461}
462
463poly ssiReadPoly(ssiInfo *D)
464{
465// < # of terms> < term1> < .....
466  int n,i,l;
467  n=ssiReadInt(D->f_read);
468  //Print("poly: terms:%d\n",n);
469  poly p;
470  int j;
471  j=0;
472  poly ret=NULL;
473  poly prev=NULL;
474  for(l=0;l<n;l++) // read n terms
475  {
476// coef,comp.exp1,..exp N
477    p=p_Init(D->r);
478    pGetCoeff(p)=ssiReadNumber(D);
479    int d;
480    SSI_BLOCK_CHLD;
481    fscanf(D->f_read,"%d",&d);
482    SSI_UNBLOCK_CHLD;
483    p_SetComp(p,d,D->r);
484    for(i=1;i<=rVar(D->r);i++)
485    {
486      SSI_BLOCK_CHLD;
487      fscanf(D->f_read,"%d",&d);
488      SSI_UNBLOCK_CHLD;
489      p_SetExp(p,i,d,D->r);
490    }
491    p_Setm(p,D->r);
492    p_Test(p,D->r);
493    if (ret==NULL) ret=p;
494    else           pNext(prev)=p;
495    prev=p;
496 }
497 return ret;
498}
499
500ideal ssiReadIdeal(ssiInfo *d)
501{
502  int n,i;
503  ideal I;
504  SSI_BLOCK_CHLD;
505  fscanf(d->f_read,"%d",&n);
506  SSI_UNBLOCK_CHLD;
507  I=idInit(n,1);
508  for(i=0;i<IDELEMS(I);i++) // read n terms
509  {
510    I->m [i]=ssiReadPoly(d);
511  }
512  return I;
513}
514
515matrix ssiReadMatrix(ssiInfo *d)
516{
517  int n,m,i,j;
518  SSI_BLOCK_CHLD;
519  fscanf(d->f_read,"%d %d",&m,&n);
520  SSI_UNBLOCK_CHLD;
521  matrix M=mpNew(m,n);
522  poly p;
523  for(int i=1;i<=MATROWS(M);i++)
524    for(int j=1;j<=MATCOLS(M);j++)
525    {
526      p=ssiReadPoly(d);
527      MATELEM(M,i,j)=p;
528    }
529  return M;
530}
531
532command ssiReadCommand(si_link l)
533{
534  ssiInfo *d=(ssiInfo*)l->data;
535  // syntax: <num ops> <operation> <op1> <op2> ....
536  command D=(command)omAlloc0(sizeof(*D));
537  int argc,op;
538  SSI_BLOCK_CHLD;
539  fscanf(d->f_read,"%d %d",&argc,&op);
540  SSI_UNBLOCK_CHLD;
541  D->argc=argc; D->op=op;
542  leftv v;
543  if (argc >0)
544  {
545    v=ssiRead1(l);
546    memcpy(&(D->arg1),v,sizeof(*v));
547    omFreeBin(v,sleftv_bin);
548  }
549  if (argc <4)
550  {
551    if (D->argc >1)
552    {
553      v=ssiRead1(l);
554      memcpy(&(D->arg2),v,sizeof(*v));
555      omFreeBin(v,sleftv_bin);
556    }
557    if (D->argc >2)
558    {
559      v=ssiRead1(l);
560      memcpy(&(D->arg3),v,sizeof(*v));
561      omFreeBin(v,sleftv_bin);
562    }
563  }
564  else
565  {
566    leftv prev=&(D->arg1);
567    argc--;
568    while(argc >0)
569    {
570      v=ssiRead1(l);
571      prev->next=v;
572      prev=v;
573      argc--;
574    }
575  }
576  return D;
577}
578
579procinfov ssiReadProc(ssiInfo *d)
580{
581  char *s=ssiReadString(d);
582  procinfov p=(procinfov)omAlloc0Bin(procinfo_bin);
583  p->language=LANG_SINGULAR;
584  p->libname=omStrDup("");
585  p->procname=omStrDup("");
586  p->data.s.body=s;
587  return p;
588}
589lists ssiReadList(si_link l)
590{
591  ssiInfo *d=(ssiInfo*)l->data;
592  int nr;
593  SSI_BLOCK_CHLD;
594  fscanf(d->f_read,"%d",&nr);
595  SSI_UNBLOCK_CHLD;
596  lists L=(lists)omAlloc(sizeof(*L));
597  L->Init(nr);
598
599  int i;
600  leftv v;
601  for(i=0;i<nr;i++)
602  {
603    v=ssiRead1(l);
604    memcpy(&(L->m[i]),v,sizeof(*v));
605    omFreeBin(v,sleftv_bin);
606  }
607  return L;
608}
609intvec* ssiReadIntvec(ssiInfo *d)
610{
611  int nr;
612  SSI_BLOCK_CHLD;
613  fscanf(d->f_read,"%d",&nr);
614  SSI_UNBLOCK_CHLD;
615  intvec *v=new intvec(nr);
616  SSI_BLOCK_CHLD;
617  for(int i=0;i<nr;i++)
618  {
619    fscanf(d->f_read,"%d",&((*v)[i]));
620  }
621  SSI_UNBLOCK_CHLD;
622  return v;
623}
624
625void ssiReadBlackbox(leftv res, si_link l)
626{
627  ssiInfo *d=(ssiInfo*)l->data;
628  int throwaway;
629  SSI_BLOCK_CHLD;
630  fscanf(d->f_read,"%d ",&throwaway);
631  SSI_UNBLOCK_CHLD;
632  char *name=ssiReadString(d);
633  int tok;
634  blackboxIsCmd(name,tok);
635  if (tok>MAX_TOK)
636  {
637    blackbox *b=getBlackboxStuff(tok);
638    res->rtyp=tok;
639    b->blackbox_deserialize(&b,&(res->data),l);
640  }
641  else
642  {
643    Werror("blackbox %s not found",name);
644  }
645}
646
647//**************************************************************************/
648
649BOOLEAN ssiOpen(si_link l, short flag, leftv u)
650{
651  const char *mode;
652  ssiInfo *d=(ssiInfo*)omAlloc0(sizeof(ssiInfo));
653  sigprocmask(SIG_SETMASK, NULL, &ssi_sigmask);
654  sigaddset(&ssi_sigmask, SIGCHLD);
655  if (flag & SI_LINK_OPEN)
656  {
657    if (l->mode[0] != '\0' && (strcmp(l->mode, "r") == 0))
658      flag = SI_LINK_READ;
659    else flag = SI_LINK_WRITE;
660  }
661
662  if (flag == SI_LINK_READ) mode = "r";
663  else if (strcmp(l->mode, "w") == 0) mode = "w";
664  else if (strcmp(l->mode, "fork") == 0) mode = "fork";
665  else if (strcmp(l->mode, "tcp") == 0) mode = "tcp";
666  else if (strcmp(l->mode, "connect") == 0) mode = "connect";
667  else mode = "a";
668
669
670  SI_LINK_SET_OPEN_P(l, flag);
671  l->data=d;
672  omFree(l->mode);
673  l->mode = omStrDup(mode);
674
675  if (l->name[0] == '\0')
676  {
677    if (strcmp(mode,"fork")==0)
678    {
679      link_list n=(link_list)omAlloc(sizeof(link_struct));
680      n->u=u;
681      n->l=l;
682      n->next=(void *)ssiToBeClosed;
683      ssiToBeClosed=n;
684
685      int pc[2];
686      int cp[2];
687      pipe(pc);
688      pipe(cp);
689      pid_t pid=fork();
690      if (pid==0) /*fork: child*/
691      {
692        link_list hh=(link_list)ssiToBeClosed->next;
693        /* we know: l is the first entry in ssiToBeClosed-list */
694        while(hh!=NULL)
695        {
696          ssiInfo *dd=(ssiInfo*)hh->l->data;
697          fclose(dd->f_read);
698          fclose(dd->f_write);
699          if (dd->r!=NULL) rKill(dd->r);
700          omFreeSize((ADDRESS)dd,(sizeof *dd));
701          hh->l->data=NULL;
702          SI_LINK_SET_CLOSE_P(hh->l);
703          link_list nn=(link_list)hh->next;
704          omFree(hh);
705          hh=nn;
706        }
707        ssiToBeClosed->next=NULL;
708        close(pc[1]); close(cp[0]);
709        d->f_read=fdopen(pc[0],"r");
710        d->fd_read=pc[0];
711        d->f_write=fdopen(cp[1],"w");
712        d->fd_write=cp[1];
713        l->data=d;
714        omFree(l->mode);
715        l->mode = omStrDup(mode);
716        singular_in_batchmode=TRUE;
717        SI_LINK_SET_RW_OPEN_P(l);
718        //myynest=0;
719        fe_fgets_stdin=fe_fgets_dummy;
720        if ((u!=NULL)&&(u->rtyp==IDHDL))
721        {
722          idhdl h=(idhdl)u->data;
723          h->lev=0;
724        }
725        loop
726        {
727          leftv h=ssiRead1(l); /*contains an exit.... */
728          if (feErrors != NULL && *feErrors != '\0')
729          {
730            // handle errors:
731            PrintS(feErrors); /* currently quite simple */
732            *feErrors = '\0';
733          }
734          ssiWrite(l,h);
735          h->CleanUp();
736          omFreeBin(h, sleftv_bin);
737        }
738        /* never reached*/
739      }
740      else if (pid>0) /*fork: parent*/
741      {
742        d->pid=pid;
743        close(pc[0]); close(cp[1]);
744        d->f_read=fdopen(cp[0],"r");
745        d->fd_read=cp[0];
746        d->f_write=fdopen(pc[1],"w");
747        d->fd_write=pc[1];
748        SI_LINK_SET_RW_OPEN_P(l);
749        d->send_quit_at_exit=1;
750      }
751      else
752      {
753        Werror("fork failed (%d)",errno);
754        l->data=NULL;
755        omFree(d);
756        return TRUE;
757      }
758    }
759    // ---------------------------------------------------------------------
760    else if (strcmp(mode,"tcp")==0)
761    {
762      int sockfd, newsockfd, portno, clilen;
763      struct sockaddr_in serv_addr, cli_addr;
764      int n;
765      sockfd = socket(AF_INET, SOCK_STREAM, 0);
766      if(sockfd < 0)
767      {
768        WerrorS("ERROR opening socket");
769        l->data=NULL;
770        omFree(d);
771        return TRUE;
772      }
773      memset((char *) &serv_addr,0, sizeof(serv_addr));
774      portno = 1025;
775      serv_addr.sin_family = AF_INET;
776      serv_addr.sin_addr.s_addr = INADDR_ANY;
777      do
778      {
779        portno++;
780        serv_addr.sin_port = htons(portno);
781        if(portno > 50000)
782        {
783          WerrorS("ERROR on binding (no free port available?)");
784          l->data=NULL;
785          omFree(d);
786          return TRUE;
787        }
788      }
789      while(bind(sockfd, (struct sockaddr *) &serv_addr, sizeof(serv_addr)) < 0);
790      Print("waiting on port %d\n", portno);mflush();
791      listen(sockfd,5);
792      newsockfd = accept(sockfd, (struct sockaddr *) &cli_addr, (socklen_t *)&clilen);
793      if(newsockfd < 0)
794      {
795        WerrorS("ERROR on accept");
796        l->data=NULL;
797        omFree(d);
798        return TRUE;
799      }
800      PrintS("client accepted\n");
801      d->fd_read = newsockfd;
802      d->fd_write = newsockfd;
803      d->f_read = fdopen(newsockfd, "r");
804      d->f_write = fdopen(newsockfd, "w");
805      SI_LINK_SET_RW_OPEN_P(l);
806      close(sockfd);
807    }
808    // no ssi-Link on stdin or stdout
809    else
810    {
811      Werror("invalid mode >>%s<< for ssi",mode);
812      l->data=NULL;
813      omFree(d);
814      return TRUE;
815    }
816  }
817  // =========================================================================
818  else /*l->name=NULL*/
819  {
820    // tcp mode
821    if(strcmp(mode,"tcp")==0)
822    {
823      int sockfd, newsockfd, portno, clilen;
824      struct sockaddr_in serv_addr, cli_addr;
825      int n;
826      sockfd = socket(AF_INET, SOCK_STREAM, 0);
827      if(sockfd < 0)
828      {
829        WerrorS("ERROR opening socket");
830        l->data=NULL;
831        omFree(d);
832        return TRUE;
833      }
834      memset((char *) &serv_addr,0, sizeof(serv_addr));
835      portno = 1025;
836      serv_addr.sin_family = AF_INET;
837      serv_addr.sin_addr.s_addr = INADDR_ANY;
838      do
839      {
840        portno++;
841        serv_addr.sin_port = htons(portno);
842        if(portno > 50000)
843        {
844          WerrorS("ERROR on binding (no free port available?)");
845          l->data=NULL;
846          return TRUE;
847        }
848      }
849      while(bind(sockfd, (struct sockaddr *) &serv_addr, sizeof(serv_addr)) < 0);
850      //Print("waiting on port %d\n", portno);mflush();
851      listen(sockfd,5);
852      char* cli_host = (char*)omAlloc(256);
853      char* path = (char*)omAlloc(1024);
854      int r = sscanf(l->name,"%255[^:]:%s",cli_host,path);
855      if(r == 0)
856      {
857        WerrorS("ERROR: no host specified");
858        l->data=NULL;
859        omFree(d);
860        omFree(path);
861        omFree(cli_host);
862        return TRUE;
863      }
864      else if(r == 1)
865      {
866        WarnS("program not specified, using /usr/local/bin/Singular");
867        strcpy(path,"/usr/local/bin/Singular");
868      }
869      char* ssh_command = (char*)omAlloc(256);
870      char* ser_host = (char*)omAlloc(64);
871      gethostname(ser_host,64);
872      sprintf(ssh_command,"ssh %s %s -q --batch --link=ssi --MPhost=%s --MPport=%d &",cli_host,path,ser_host,portno);
873      //Print("client on %s started:%s\n",cli_host,path);
874      omFree(path);
875      omFree(cli_host);
876      if (TEST_OPT_PROT) { Print("running >>%s<<\n",ssh_command); }
877      system(ssh_command);
878      omFree(ssh_command);
879      omFree(ser_host);
880      clilen = sizeof(cli_addr);
881      newsockfd = accept(sockfd, (struct sockaddr *) &cli_addr, (socklen_t *)&clilen);
882      if(newsockfd < 0)
883      {
884        WerrorS("ERROR on accept");
885        l->data=NULL;
886        omFree(d);
887        return TRUE;
888      }
889      //PrintS("client accepted\n");
890      d->fd_read = newsockfd;
891      d->fd_write = newsockfd;
892      d->f_read = fdopen(newsockfd, "r");
893      d->f_write = fdopen(newsockfd, "w");
894      SI_LINK_SET_RW_OPEN_P(l);
895      d->send_quit_at_exit=1;
896      close(sockfd);
897      fprintf(d->f_write,"98 %d %d %u %u\n",SSI_VERSION,MAX_TOK,test,verbose);
898    }
899    // ----------------------------------------------------------------------
900    else if(strcmp(mode,"connect")==0)
901    {
902      char* host = (char*)omAlloc(256);
903      int sockfd, portno, n;
904      struct sockaddr_in serv_addr;
905      struct hostent *server;
906
907      sscanf(l->name,"%255[^:]:%d",host,&portno);
908      //Print("connect to host %s, port %d\n",host,portno);mflush();
909      if (portno!=0)
910      {
911        sockfd = socket(AF_INET, SOCK_STREAM, 0);
912        if (sockfd < 0) { WerrorS("ERROR opening socket"); return TRUE; }
913        server = gethostbyname(host);
914        if (server == NULL) {  WerrorS("ERROR, no such host");  return TRUE; }
915        memset((char *) &serv_addr, 0, sizeof(serv_addr));
916        serv_addr.sin_family = AF_INET;
917        memcpy((char *)&serv_addr.sin_addr.s_addr,
918              (char *)server->h_addr,
919              server->h_length);
920        serv_addr.sin_port = htons(portno);
921        if (connect(sockfd,(sockaddr*)&serv_addr,sizeof(serv_addr)) < 0)
922        { WerrorS("ERROR connecting"); return TRUE; }
923        //PrintS("connected\n");mflush();
924        d->f_read=fdopen(sockfd,"r");
925        d->fd_read=sockfd;
926        d->f_write=fdopen(sockfd,"w");
927        d->fd_write=sockfd;
928        SI_LINK_SET_RW_OPEN_P(l);
929        omFree(host);
930      }
931      else
932      {
933        l->data=NULL;
934        omFree(d);
935        return TRUE;
936      }
937    }
938    // ======================================================================
939    else
940    {
941      // normal link to a file
942      FILE *outfile;
943      char *filename=l->name;
944
945      if(filename[0]=='>')
946      {
947        if (filename[1]=='>')
948        {
949          filename+=2;
950          mode = "a";
951        }
952        else
953        {
954          filename++;
955          mode="w";
956        }
957      }
958      outfile=myfopen(filename,mode);
959      if (outfile!=NULL)
960      {
961        if (strcmp(l->mode,"r")==0) d->f_read = outfile;
962        else
963        {
964          d->f_write = outfile;
965          fprintf(d->f_write,"98 %d %d %u %u\n",SSI_VERSION,MAX_TOK,test,verbose);
966        }
967      }
968      else
969      {
970        omFree(d);
971        l->data=NULL;
972        return TRUE;
973      }
974    }
975  }
976
977  return FALSE;
978}
979
980//**************************************************************************/
981BOOLEAN ssiClose(si_link l)
982{
983  ssiInfo *d = (ssiInfo *)l->data;
984  if (d!=NULL)
985  {
986    if (d->r!=NULL) rKill(d->r);
987    if ((strcmp(l->mode,"tcp")==0)
988    || (strcmp(l->mode,"fork")==0))
989    {
990      link_list hh=ssiToBeClosed;
991      if (hh!=NULL)
992      {
993        if (hh->l==l)
994        {
995           ssiToBeClosed=(link_list)hh->next;
996           omFreeSize(hh,sizeof(link_struct));
997        }
998        else while(hh!=NULL)
999        {
1000          link_list hhh=(link_list)hh->next;
1001          if (hhh->l==l)
1002          {
1003            hh->next=hhh->next;
1004            omFreeSize(hhh,sizeof(link_struct));
1005            break;
1006          }
1007          hh=(link_list)hh->next;
1008        }
1009      }
1010      if (d->send_quit_at_exit)
1011      {
1012        fprintf(d->f_write,"99\n");fflush(d->f_write);
1013      }
1014    }
1015    if (d->f_read!=NULL) fclose(d->f_read);
1016    if (d->f_write!=NULL) fclose(d->f_write);
1017    if (d->pid!=0)
1018    {
1019      int status;
1020      if (kill(d->pid,15)!=0)
1021      {
1022         waitpid(d->pid,&status,WNOHANG);
1023      }
1024    }
1025    omFreeSize((ADDRESS)d,(sizeof *d));
1026  }
1027  l->data=NULL;
1028  SI_LINK_SET_CLOSE_P(l);
1029  return FALSE;
1030}
1031
1032//**************************************************************************/
1033leftv ssiRead1(si_link l)
1034{
1035  ssiInfo *d = (ssiInfo *)l->data;
1036  d->ungetc_buf='\0';
1037  leftv res=(leftv)omAlloc0(sizeof(sleftv));
1038  int t=0;
1039  SSI_BLOCK_CHLD;
1040  fscanf(d->f_read,"%d",&t);
1041  SSI_UNBLOCK_CHLD;
1042  //Print("got type %d\n",t);
1043  switch(t)
1044  {
1045    case 1:res->rtyp=INT_CMD;
1046           res->data=(char *)ssiReadInt(d->f_read);
1047           break;
1048    case 2:res->rtyp=STRING_CMD;
1049           res->data=(char *)ssiReadString(d);
1050           break;
1051    case 3:res->rtyp=NUMBER_CMD;
1052           res->data=(char *)ssiReadNumber(d);
1053           break;
1054    case 4:res->rtyp=BIGINT_CMD;
1055           res->data=(char *)ssiReadBigInt(d);
1056           break;
1057    case 15:
1058    case 5:{
1059             d->r=ssiReadRing(d);
1060             d->r->ref++;
1061             res->rtyp=RING_CMD;
1062             res->data=(char*)d->r;
1063             // we are in the top-level, so set the basering to d->r:
1064             ssiSetCurrRing(d->r);
1065             if (t==15) return ssiRead1(l);
1066           }
1067           break;
1068    case 6:res->rtyp=POLY_CMD;
1069           if (d->r==NULL) goto no_ring;
1070           res->data=(char*)ssiReadPoly(d);
1071           break;
1072    case 7:res->rtyp=IDEAL_CMD;
1073           if (d->r==NULL) goto no_ring;
1074           res->data=(char*)ssiReadIdeal(d);
1075           break;
1076    case 8:res->rtyp=MATRIX_CMD;
1077           if (d->r==NULL) goto no_ring;
1078           res->data=(char*)ssiReadMatrix(d);
1079           break;
1080    case 9:res->rtyp=VECTOR_CMD;
1081           if (d->r==NULL) goto no_ring;
1082           res->data=(char*)ssiReadPoly(d);
1083           break;
1084    case 10:res->rtyp=MODUL_CMD;
1085           if (d->r==NULL) goto no_ring;
1086           res->data=(char*)ssiReadIdeal(d);
1087           break;
1088    case 11:
1089           {
1090             res->rtyp=COMMAND;
1091             res->data=ssiReadCommand(l);
1092             int nok=res->Eval();
1093             if (nok) WerrorS("error in eval");
1094             break;
1095           }
1096    case 12: /*DEF_CMD*/
1097           {
1098             res->rtyp=0;
1099             res->name=(char *)ssiReadString(d);
1100             int nok=res->Eval();
1101             if (nok) WerrorS("error in name lookup");
1102             break;
1103           }
1104    case 13: res->rtyp=PROC_CMD;
1105             res->data=ssiReadProc(d);
1106             break;
1107    case 14: res->rtyp=LIST_CMD;
1108             res->data=ssiReadList(l);
1109             break;
1110    case 16: res->rtyp=NONE; res->data=NULL;
1111             break;
1112    case 17: res->rtyp=INTVEC_CMD;
1113             res->data=ssiReadIntvec(d);
1114             break;
1115    case 20: ssiReadBlackbox(res,l);
1116             break;
1117    // ------------
1118    case 98: // version
1119             {
1120                int n98_v,n98_m;
1121                BITSET n98_o1,n98_o2;
1122                SSI_BLOCK_CHLD;
1123                fscanf(d->f_read,"%d %d %u %u\n",&n98_v,&n98_m,&n98_o1,&n98_o2);
1124                SSI_UNBLOCK_CHLD;
1125                if ((n98_v!=SSI_VERSION) ||(n98_m!=MAX_TOK))
1126                {
1127                  Print("incompatible versions of ssi: %d/%d vs %d/%d",
1128                                  SSI_VERSION,MAX_TOK,n98_v,n98_m);
1129                }
1130                #ifndef NDEBUG
1131                if (TEST_OPT_DEBUG)
1132                  Print("// opening ssi-%d, MAX_TOK=%d\n",n98_v,n98_m);
1133                #endif
1134                test=n98_o1;
1135                verbose=n98_o2;
1136                return ssiRead1(l);
1137             }
1138    case 99: ssiClose(l); exit(0);
1139    case 0: if (feof(d->f_read))
1140            {
1141              ssiClose(l);
1142              res->rtyp=DEF_CMD;
1143              break;
1144            }
1145    default: Werror("not implemented (t:%d)",t);
1146             omFreeSize(res,sizeof(sleftv));
1147             res=NULL;
1148             break;
1149  }
1150  return res;
1151no_ring: WerrorS("no ring");
1152  omFreeSize(res,sizeof(sleftv));
1153  return NULL;
1154}
1155//**************************************************************************/
1156BOOLEAN ssiWrite(si_link l, leftv data)
1157{
1158  if(SI_LINK_W_OPEN_P(l)==0) slOpen(l,SI_LINK_OPEN|SI_LINK_WRITE,NULL);
1159  ssiInfo *d = (ssiInfo *)l->data;
1160  d->level++;
1161  //FILE *fich=d->f;
1162  while (data!=NULL)
1163  {
1164    int tt=data->Typ();
1165    void *dd=data->Data();
1166
1167    switch(tt /*data->Typ()*/)
1168    {
1169          case NONE/* nothing*/:fprintf(d->f_write,"16 ");
1170                          break;
1171          case STRING_CMD: fprintf(d->f_write,"2 ");
1172                           ssiWriteString(d,(char *)dd);
1173                           break;
1174          case INT_CMD: fprintf(d->f_write,"1 ");
1175                        ssiWriteInt(d,(int)(long)dd);
1176                        break;
1177          case BIGINT_CMD:fprintf(d->f_write,"4 ");
1178                        ssiWriteBigInt(d,(number)dd);
1179                        break;
1180          case NUMBER_CMD:
1181                          if (d->r!=currRing)
1182                          {
1183                            fprintf(d->f_write,"15 ");
1184                            ssiWriteRing(d,currRing);
1185                            if (d->level<=1) fprintf(d->f_write,"\n");
1186                          }
1187                          fprintf(d->f_write,"3 ");
1188                          ssiWriteNumber(d,(number)dd);
1189                        break;
1190          case RING_CMD:fprintf(d->f_write,"5 ");
1191                        ssiWriteRing(d,(ring)dd);
1192                        break;
1193          case POLY_CMD:
1194          case VECTOR_CMD:
1195                        if (d->r!=currRing)
1196                        {
1197                          fprintf(d->f_write,"15 ");
1198                          ssiWriteRing(d,currRing);
1199                          if (d->level<=1) fprintf(d->f_write,"\n");
1200                        }
1201                        if(tt==POLY_CMD) fprintf(d->f_write,"6 ");
1202                        else             fprintf(d->f_write,"9 ");
1203                        ssiWritePoly(d,tt,(poly)dd);
1204                        break;
1205          case IDEAL_CMD:
1206          case MODUL_CMD:
1207          case MATRIX_CMD:
1208                        if (d->r!=currRing)
1209                        {
1210                          fprintf(d->f_write,"15 ");
1211                          ssiWriteRing(d,currRing);
1212                          if (d->level<=1) fprintf(d->f_write,"\n");
1213                        }
1214                        if(tt==IDEAL_CMD)       fprintf(d->f_write,"7 ");
1215                        else if(tt==MATRIX_CMD) fprintf(d->f_write,"8 ");
1216                        else                    fprintf(d->f_write,"10 ");
1217                        ssiWriteIdeal(d,tt,(ideal)dd);
1218                        break;
1219          case COMMAND:
1220                   fprintf(d->f_write,"11 ");
1221                   ssiWriteCommand(l,(command)dd);
1222                   break;
1223          case DEF_CMD: /* not evaluated stuff in quotes */
1224                   fprintf(d->f_write,"12 ");
1225                   ssiWriteString(d,data->Name());
1226                   break;
1227          case PROC_CMD:
1228                   fprintf(d->f_write,"13 ");
1229                   ssiWriteProc(d,(procinfov)dd);
1230                   break;
1231          case LIST_CMD:
1232                   fprintf(d->f_write,"14 ");
1233                   ssiWriteList(l,(lists)dd);
1234                   break;
1235          case INTVEC_CMD:
1236                   fprintf(d->f_write,"17 ");
1237                   ssiWriteIntvec(d,(intvec *)dd);
1238                   break;
1239          default:
1240            if (tt>MAX_TOK)
1241            {
1242              blackbox *b=getBlackboxStuff(tt);
1243              fprintf(d->f_write,"20 ");
1244              b->blackbox_serialize(b,dd,l);
1245            }
1246            else
1247            {
1248              Werror("not implemented (t:%d, rtyp:%d)",tt, data->rtyp);
1249              d->level=0;
1250              return TRUE;
1251            }
1252            break;
1253    }
1254    if (d->level<=1) { fprintf(d->f_write,"\n"); fflush(d->f_write); }
1255    data=data->next;
1256  }
1257  d->level--;
1258  return FALSE;
1259}
1260
1261si_link_extension slInitSsiExtension(si_link_extension s)
1262{
1263  s->Open=ssiOpen;
1264  s->Close=ssiClose;
1265  s->Kill=ssiClose;
1266  s->Read=ssiRead1;
1267  s->Read2=(slRead2Proc)NULL;
1268  s->Write=ssiWrite;
1269
1270  s->Status=slStatusSsi;
1271  s->type="ssi";
1272  return s;
1273}
1274
1275const char* slStatusSsi(si_link l, const char* request)
1276{
1277  ssiInfo *d=(ssiInfo*)l->data;
1278  if (d==NULL) return "not open";
1279  if (((strcmp(l->mode,"fork")==0)||(strcmp(l->mode,"tcp")==0))
1280  && (strcmp(request, "read") == 0))
1281  {
1282    fd_set  mask, fdmask;
1283    struct timeval wt;
1284    if (d->ungetc_buf) return "ready";
1285    loop
1286    {
1287      /* Don't block. Return socket status immediately. */
1288      wt.tv_sec  = 0;
1289      wt.tv_usec = 0;
1290
1291      FD_ZERO(&mask);
1292      FD_SET(d->fd_read, &mask);
1293      //Print("test fd %d\n",d->fd_read);
1294    /* check with select: chars waiting: no -> not ready */
1295      switch (select(d->fd_read+1, &mask, NULL, NULL, &wt))
1296      {
1297        case 0: /* not ready */ return "not ready";
1298        case -1: /*error*/      return "error";
1299        case 1: /*ready ? */    break;
1300      }
1301    /* yes: read 1 char*/
1302    /* if \n, check again with select else ungetc(c), ready*/
1303      int c=fgetc(d->f_read);
1304      //Print("try c=%d\n",c);
1305      if (c== -1) return "eof"; /* eof or error */
1306      else if (isdigit(c))
1307      { ungetc(c,d->f_read); d->ungetc_buf='\1'; return "ready"; }
1308      else if (c>' ')
1309      {
1310        Werror("unknown char in ssiLink(%d)",c);
1311        return "error";
1312      }
1313      /* else: next char */
1314    }
1315  }
1316  else if (strcmp(request, "read") == 0)
1317  {
1318    if (SI_LINK_R_OPEN_P(l) && (!feof(d->f_read))) return "ready";
1319    else return "not ready";
1320  }
1321  else if (strcmp(request, "write") == 0)
1322  {
1323    if (SI_LINK_W_OPEN_P(l)) return "ready";
1324    else return "not ready";
1325  }
1326  else return "unknown status request";
1327}
1328
1329int slStatusSsiL(lists L, int timeout)
1330{
1331// input: L: a list with links of type
1332//           ssi-fork, ssi-tcp, MPtcp-fork or MPtcp-launch.
1333//           Note: Not every entry in L must be set.
1334//        timeout: timeout for select in micro-seconds
1335//           or -1 for infinity
1336//           or 0 for polling
1337// returns: ERROR (via Werror): L has wrong elements or link not open
1338//           -2: select returns an error
1339//           -1: the read state of all links is eof
1340//           0:  timeout (or polling): none ready,
1341//           i>0: (at least) L[i] is ready
1342  si_link l;
1343  ssiInfo *d;
1344  #ifdef HAVE_MPSR
1345  MP_Link_pt dd;
1346  #endif
1347  int d_fd;
1348  fd_set  mask, fdmask;
1349  FD_ZERO(&fdmask);
1350  FD_ZERO(&mask);
1351  int max_fd=0; /* 1 + max fd in fd_set */
1352
1353  /* timeout */
1354  #ifdef HAVE_PSELECT
1355  struct timespec wt;
1356  struct timespec *wt_ptr=&wt;
1357  #else
1358  struct timeval wt;
1359  struct timeval *wt_ptr=&wt;
1360  #endif
1361  int startingtime = getRTimer()/TIMER_RESOLUTION;  // in seconds
1362  if (timeout== -1)
1363  {
1364    wt_ptr=NULL;
1365  }
1366  else
1367  {
1368    wt.tv_sec  = timeout / 1000000;
1369  #ifdef HAVE_PSELECT
1370    wt.tv_nsec = 1000 * (timeout % 1000000);
1371  #else
1372    wt.tv_usec = timeout % 1000000;
1373  #endif
1374  }
1375
1376  /* signal mask for pselect() */
1377  sigset_t sigmask;
1378  if(sigprocmask(SIG_SETMASK, NULL, &sigmask) < 0)
1379  {
1380    WerrorS("error in sigprocmask()");
1381    return -2;
1382  }
1383  if(sigaddset(&sigmask, SIGCHLD) < 0)
1384  {
1385    WerrorS("error in sigaddset()");
1386    return -2;
1387  }
1388
1389  /* auxiliary variables */
1390  int i;
1391  int j;
1392  int k;
1393  int s;
1394  char fdmaskempty;
1395
1396  /* check the links and fill in fdmask */
1397  for(i=L->nr; i>=0; i--)
1398  {
1399    if (L->m[i].Typ()!=DEF_CMD)
1400    {
1401      if (L->m[i].Typ()!=LINK_CMD)
1402      { WerrorS("all elements must be of type link"); return -2;}
1403      l=(si_link)L->m[i].Data();
1404      if(SI_LINK_OPEN_P(l)==0)
1405      { WerrorS("all links must be open"); return -2;}
1406      if (((strcmp(l->m->type,"ssi")!=0) && (strcmp(l->m->type,"MPtcp")!=0))
1407      || ((strcmp(l->mode,"fork")!=0) && (strcmp(l->mode,"tcp")!=0)
1408        && (strcmp(l->mode,"launch")!=0)))
1409      {
1410        WerrorS("all links must be of type ssi:fork, ssi:tcp, MPtcp:fork\n");
1411        WerrorS("or MPtcp:launch");
1412        return -2;
1413      }
1414    #ifdef HAVE_MPSR
1415      if (strcmp(l->m->type,"ssi")==0)
1416      {
1417        d=(ssiInfo*)l->data;
1418        d_fd=d->fd_read;
1419      }
1420      else
1421      {
1422        dd=(MP_Link_pt)l->data;
1423        d_fd=((MP_TCP_t *)dd->transp.private1)->sock;
1424      }
1425    #else
1426      d=(ssiInfo*)l->data;
1427      d_fd=d->fd_read;
1428    #endif
1429      FD_SET(d_fd, &fdmask);
1430      if (d_fd > max_fd) max_fd=d_fd;
1431    }
1432  }
1433  max_fd++;
1434
1435do_select:
1436  /* copy fdmask to mask */
1437  FD_ZERO(&mask);
1438  for(k = 0; k < max_fd; k++)
1439  {
1440    if(FD_ISSET(k, &fdmask))
1441    {
1442      FD_SET(k, &mask);
1443    }
1444  }
1445
1446  /* check with select: chars waiting: no -> not ready */
1447  #ifdef HAVE_PSELECT
1448  s = pselect(max_fd, &mask, NULL, NULL, wt_ptr, &sigmask);
1449  #else
1450  SSI_BLOCK_CHLD;
1451  s = select(max_fd, &mask, NULL, NULL, wt_ptr);
1452  SSI_UNBLOCK_CHLD;
1453  #endif
1454
1455  if (s==-1)
1456  {
1457    WerrorS("error in select call");
1458    return -2; /*error*/
1459  }
1460  if (s==0)
1461  {
1462    return 0; /*poll: not ready */
1463  }
1464  else /* s>0, at least one ready  (the number of fd which are ready is s)*/
1465  {
1466    j=0;
1467    while (j<=max_fd) { if (FD_ISSET(j,&mask)) break; j++; }
1468    for(i=L->nr; i>=0; i--)
1469    {
1470      if (L->m[i].rtyp==LINK_CMD)
1471      {
1472        l=(si_link)L->m[i].Data();
1473        #ifdef HAVE_MPSR
1474        if (strcmp(l->m->type,"ssi")!=0)
1475        {
1476          // for MP links, return here:
1477          dd=(MP_Link_pt)l->data;
1478          d_fd=((MP_TCP_t *)dd->transp.private1)->sock;
1479          if(j==d_fd) return i+1;
1480        }
1481        else
1482        {
1483          d=(ssiInfo*)l->data;
1484          d_fd=d->fd_read;
1485          if(j==d_fd) break;
1486        }
1487        #else
1488        d=(ssiInfo*)l->data;
1489        d_fd=d->fd_read;
1490        if(j==d_fd) break;
1491        #endif
1492      }
1493    }
1494    // only ssi links:
1495    if (d->ungetc_buf) return i+1;
1496    loop
1497    {
1498      /* yes: read 1 char*/
1499      /* if \n, check again with select else ungetc(c), ready*/
1500      /* setting: d: current ssiInfo, j current fd, i current entry in L*/
1501      int c=fgetc(d->f_read);
1502      //Print("try c=%d\n",c);
1503      if (c== -1) /* eof */
1504      {
1505        FD_CLR(j,&fdmask);
1506        fdmaskempty = 1;
1507        for(k = 0; k < max_fd; k++)
1508        {
1509          if(FD_ISSET(k, &fdmask))
1510          {
1511            fdmaskempty = 0;
1512            break;
1513          }
1514        }
1515        if(fdmaskempty)
1516        {
1517          return -1;
1518        }
1519        if(timeout != -1)
1520        {
1521          timeout = si_max(0,
1522               timeout - 1000000*(getRTimer()/TIMER_RESOLUTION - startingtime));
1523          wt.tv_sec  = timeout / 1000000;
1524          #ifdef HAVE_PSELECT
1525          wt.tv_nsec = 1000 * (timeout % 1000000);
1526          #else
1527          wt.tv_usec = (timeout % 1000000);
1528          #endif
1529        }
1530        goto do_select;
1531      }
1532
1533      else if (isdigit(c))
1534      { ungetc(c,d->f_read); d->ungetc_buf='\1'; return i+1; }
1535      else if (c>' ')
1536      {
1537        Werror("unknown char in ssiLink(%d)",c);
1538        return -2;
1539      }
1540      /* else: next char */
1541    }
1542  }
1543}
1544
1545int ssiBatch(const char *host, const char * port)
1546/* return 0 on success, >0 else*/
1547{
1548  si_link l=(si_link) omAlloc0Bin(sip_link_bin);
1549  char *buf=(char*)omAlloc(256);
1550  sprintf(buf,"ssi:connect %s:%s",host,port);
1551  slInit(l, buf);
1552  slOpen(l,SI_LINK_OPEN,NULL);
1553  SI_LINK_SET_RW_OPEN_P(l);
1554
1555  idhdl id = enterid(omStrDup("link_ll"), 0, LINK_CMD, &IDROOT, FALSE);
1556  IDLINK(id) = l;
1557
1558  loop
1559  {
1560    leftv h=ssiRead1(l); /*contains an exit.... */
1561    if (feErrors != NULL && *feErrors != '\0')
1562    {
1563      // handle errors:
1564      PrintS(feErrors); /* currently quite simple */
1565      *feErrors = '\0';
1566    }
1567    ssiWrite(l,h);
1568    h->CleanUp();
1569    omFreeBin(h, sleftv_bin);
1570  }
1571  /* never reached*/
1572  exit(0);
1573}
1574
1575// ----------------------------------------------------------------
1576// format
1577// 1 int %d
1578// 2 string <len> %s
1579// 3 number
1580// 4 bigint 4 %d or 3 <mpz_t>
1581// 5 ring
1582// 6 poly
1583// 7 ideal
1584// 8 matrix
1585// 9 vector
1586// 10 module
1587// 11 command
1588// 12 def <len> %s
1589// 13 proc <len> %s
1590// 14 list %d <elem1> ....
1591// 15 setring .......
1592// 16 nothing
1593// 17 intvec <len> ...
1594//
1595// 20 blackbox <name> ...
1596//
1597// 98: verify version: <ssi-version> <MAX_TOK> <OPT1> <OPT2>
1598// 99: quit Singular
Note: See TracBrowser for help on using the repository browser.