source: git/Singular/links/ssiLink.cc @ d3063fd

spielwiese
Last change on this file since d3063fd was d3063fd, checked in by Hans Schoenemann <hannes@…>, 10 years ago
fix: wait may be undefined
  • Property mode set to 100644
File size: 50.7 KB
Line 
1/****************************************
2 * Computer Algebra System SINGULAR     *
3 ****************************************/
4/***************************************************************
5 * File:    ssiLink.h
6 *  Purpose: declaration of sl_link routines for ssi
7 ***************************************************************/
8#include <stdio.h>
9#include <fcntl.h>
10#include <errno.h>
11#include <unistd.h>
12#include <sys/types.h>
13#include <signal.h>
14#include <sys/types.h>          /* for portability */
15#include <sys/select.h>
16#include <sys/socket.h>
17#include <netinet/in.h>
18#include <ctype.h>   /*for isdigit*/
19#include <netdb.h>
20#include <sys/wait.h>
21#include <time.h>
22
23#include <kernel/mod2.h>
24#include <Singular/si_signals.h>
25// #include "mod2.h"
26
27#include <Singular/tok.h>
28#include <Singular/ipid.h>
29#include <Singular/ipshell.h>
30#include <omalloc/omalloc.h>
31#include <libpolys/polys/monomials/ring.h>
32#include <libpolys/polys/matpol.h>
33#include <libpolys/polys/simpleideals.h>
34#include <libpolys/polys/monomials/p_polys.h>
35#include <libpolys/coeffs/longrat.h>
36#include <libpolys/misc/intvec.h>
37#include <libpolys/coeffs/bigintmat.h>
38#include <libpolys/misc/options.h>
39#include <kernel/timer.h>
40#include <Singular/rlimit.h>
41#include <Singular/subexpr.h>
42#include <Singular/links/silink.h>
43#include <Singular/cntrlc.h>
44#include <Singular/lists.h>
45#include <Singular/blackbox.h>
46#include <Singular/links/s_buff.h>
47#include <Singular/links/ssiLink.h>
48
49#ifdef HAVE_SIMPLEIPC
50#include <Singular/links/simpleipc.h>
51#endif
52
53#include <Singular/si_signals.h>
54
55#define SSI_VERSION 5
56
57// 64 bit version:
58//#if SIZEOF_LONG == 8
59#if 0
60#define MAX_NUM_SIZE 60
61#define POW_2_28 (1L<<60)
62#define LONG long
63#else
64// 32 bit version:
65#define MAX_NUM_SIZE 28
66#define POW_2_28 (1L<<28)
67#define LONG int
68#endif
69typedef struct
70{
71  s_buff f_read;
72  FILE *f_write;
73  ring r;
74  pid_t pid; /* only valid for fork/tcp mode*/
75  int fd_read,fd_write; /* only valid for fork/tcp mode*/
76  char level;
77  char send_quit_at_exit;
78  char quit_sent;
79
80} ssiInfo;
81
82link_list ssiToBeClosed=NULL;
83volatile BOOLEAN ssiToBeClosed_inactive=TRUE;
84
85// the helper functions:
86void ssiSetCurrRing(const ring r)
87{
88  if (!rEqual(r,currRing,1))
89  {
90    char name[20];
91    int nr=0;
92    do
93    { sprintf(name,"ssiRing%d",nr); nr++; }
94    while(IDROOT->get(name, 0)!=NULL);
95    idhdl h=enterid(omStrDup(name),0,RING_CMD,&IDROOT,FALSE);
96    IDRING(h)=r;
97    r->ref++;
98    rSetHdl(h);
99  }
100}
101// the implementation of the functions:
102void ssiWriteInt(ssiInfo *d,const int i)
103{
104  fprintf(d->f_write,"%d ",i);
105  //if (d->f_debug!=NULL) fprintf(d->f_debug,"int: %d ",i);
106}
107
108void ssiWriteString(ssiInfo *d,const char *s)
109{
110  fprintf(d->f_write,"%d %s ",(int)strlen(s),s);
111  //if (d->f_debug!=NULL) fprintf(d->f_debug,"stringi: %d \"%s\" ",strlen(s),s);
112}
113
114
115void ssiWriteBigInt(const ssiInfo *d, const number n)
116{
117  // syntax is as follows:
118  // case 2 Q:     3 4 <int>
119  //        or     3 3 <mpz_t nominator>
120  if(SR_HDL(n) & SR_INT)
121  {
122    fprintf(d->f_write,"4 %ld ",SR_TO_INT(n));
123    //if (d->f_debug!=NULL) fprintf(d->f_debug,"bigint: short \"%ld\" ",SR_TO_INT(n));
124  }
125  else if (n->s==3)
126  {
127    fputs("3 ",d->f_write);
128    mpz_out_str(d->f_write,10,n->z);
129    fputc(' ',d->f_write);
130    //gmp_fprintf(d->f_write,"3 %Zd ",n->z);
131    //if (d->f_debug!=NULL) gmp_fprintf(d->f_debug,"bigint: gmp \"%Zd\" ",n->z);
132  }
133  else WerrorS("illiegal bigint");
134}
135
136void ssiWriteNumber(const ssiInfo *d, const number n)
137{
138  // syntax is as follows:
139  // case 1 Z/p:   3 <int>
140  // case 2 Q:     3 4 <int>
141  //        or     3 0 <mpz_t nominator> <mpz_t denominator>
142  //        or     3 1  dto.
143  //        or     3 3 <mpz_t nominator>
144  //        or     3 5 <mpz_t raw nom.> <mpz_t raw denom.>
145  //        or     3 6 <mpz_t raw nom.> <mpz_t raw denom.>
146  //        or     3 7 <mpz_t raw nom.>
147  if(rField_is_Zp(d->r))
148  {
149    fprintf(d->f_write,"%d ",(int)(long)n);
150    //if (d->f_debug!=NULL) fprintf(d->f_debug,"number: \"%ld\" ",(int)(long)n);
151  }
152  else if (rField_is_Q(d->r))
153  {
154    if(SR_HDL(n) & SR_INT)
155    {
156      #if SIZEOF_LONG == 4
157      fprintf(d->f_write,"4 %ld ",SR_TO_INT(n));
158      #else
159      long nn=SR_TO_INT(n);
160      if ((nn<POW_2_28)||(nn>= -POW_2_28))
161        fprintf(d->f_write,"4 %ld ",nn);
162      else
163      {
164        mpz_t tmp;
165        mpz_init_set_si(tmp,nn);
166        fputs("8 ",d->f_write);
167        mpz_out_str (d->f_write,32, tmp);
168        fputc(' ',d->f_write);
169        mpz_clear(tmp);
170      }
171      #endif
172      //if (d->f_debug!=NULL) fprintf(d->f_debug,"number: short \"%ld\" ",SR_TO_INT(n));
173    }
174    else if (n->s<2)
175    {
176      //gmp_fprintf(d->f_write,"%d %Zd %Zd ",n->s,n->z,n->n);
177      fprintf(d->f_write,"%d ",n->s+5);
178      mpz_out_str (d->f_write,32, n->z);
179      fputc(' ',d->f_write);
180      mpz_out_str (d->f_write,32, n->n);
181      fputc(' ',d->f_write);
182
183      //if (d->f_debug!=NULL) gmp_fprintf(d->f_debug,"number: s=%d gmp/gmp \"%Zd %Zd\" ",n->s,n->z,n->n);
184    }
185    else /*n->s==3*/
186    {
187      //gmp_fprintf(d->f_write,"3 %Zd ",n->z);
188      fputs("8 ",d->f_write);
189      mpz_out_str (d->f_write,32, n->z);
190      fputc(' ',d->f_write);
191
192      //if (d->f_debug!=NULL) gmp_fprintf(d->f_debug,"number: gmp \"%Zd\" ",n->z);
193    }
194  }
195  else WerrorS("coeff field not implemented");
196}
197
198void ssiWriteRing(ssiInfo *d,const ring r)
199{
200  /* 5 <ch> <N> <l1> <v1> ...<lN> <vN> <number of orderings> <ord1> <block0_1> <block1_1> .... */
201  if (d->r!=NULL) rKill(d->r);
202  d->r=r;
203  if (r!=NULL)
204  {
205    /*d->*/r->ref++;
206    fprintf(d->f_write,"%d %d ",n_GetChar(r->cf),r->N);
207
208    int i;
209    for(i=0;i<r->N;i++)
210    {
211      fprintf(d->f_write,"%d %s ",(int)strlen(r->names[i]),r->names[i]);
212    }
213    /* number of orderings:*/
214    i=0;
215    // remember dummy ring: everything 0:
216    if (r->order!=NULL) while (r->order[i]!=0) i++;
217    fprintf(d->f_write,"%d ",i);
218    /* each ordering block: */
219    i=0;
220    if (r->order!=NULL) while(r->order[i]!=0)
221    {
222      fprintf(d->f_write,"%d %d %d ",r->order[i],r->block0[i], r->block1[i]);
223      switch(r->order[i])
224      {
225        case ringorder_a:
226        case ringorder_wp:
227        case ringorder_Wp:
228        case ringorder_ws:
229        case ringorder_Ws:
230        case ringorder_aa:
231        {
232          int ii;
233          for(ii=r->block0[i];ii<=r->block1[i];ii++)
234            fprintf(d->f_write,"%d ",r->wvhdl[i][ii-r->block0[i]]);
235        }
236        break;
237
238        case ringorder_a64:
239        case ringorder_M:
240        case ringorder_L:
241        case ringorder_IS:
242          Werror("ring oder not implemented for ssi:%d",r->order[i]);
243          break;
244
245        default: break;
246      }
247      i++;
248    }
249  }
250  else /* dummy ring r==NULL*/
251  {
252    fprintf(d->f_write,"0 0 0 "/*,r->ch,r->N, blocks*/);
253  }
254}
255
256void ssiWritePoly(ssiInfo *d, int typ, poly p)
257{
258  fprintf(d->f_write,"%d ",pLength(p));//number of terms
259  int i;
260
261  while(p!=NULL)
262  {
263    ssiWriteNumber(d,pGetCoeff(p));
264    //nWrite(fich,pGetCoeff(p));
265    fprintf(d->f_write,"%ld ",p_GetComp(p,d->r));//component
266
267    for(int j=1;j<=rVar(d->r);j++)
268    {
269      fprintf(d->f_write,"%ld ",p_GetExp(p,j,d->r ));//x^j
270    }
271    pIter(p);
272  }
273}
274
275void ssiWriteIdeal(ssiInfo *d, int typ,ideal I)
276{
277   // syntax: 7 # of elements <poly 1> <poly2>.....
278   // syntax: 8 <rows> <cols> <poly 1> <poly2>.....
279   matrix M=(matrix)I;
280   int mn;
281   if (typ==MATRIX_CMD)
282   {
283     mn=MATROWS(M)*MATCOLS(M);
284     fprintf(d->f_write,"%d %d ", MATROWS(M),MATCOLS(M));
285   }
286   else
287   {
288     mn=IDELEMS(I);
289     fprintf(d->f_write,"%d ",IDELEMS(I));
290   }
291
292   int i;
293   int tt;
294   if (typ==MODUL_CMD) tt=VECTOR_CMD;
295   else                tt=POLY_CMD;
296
297   for(i=0;i<mn;i++)
298   {
299     ssiWritePoly(d,tt,I->m[i]);
300   }
301}
302
303void ssiWriteCommand(si_link l, command D)
304{
305  ssiInfo *d=(ssiInfo*)l->data;
306  // syntax: <num ops> <operation> <op1> <op2> ....
307  fprintf(d->f_write,"%d %d ",D->argc,D->op);
308  if (D->argc >0) ssiWrite(l, &(D->arg1));
309  if (D->argc < 4)
310  {
311    if (D->argc >1) ssiWrite(l, &(D->arg2));
312    if (D->argc >2) ssiWrite(l, &(D->arg3));
313  }
314}
315
316void ssiWriteProc(ssiInfo *d,procinfov p)
317{
318  if (p->data.s.body==NULL)
319    iiGetLibProcBuffer(p);
320  if (p->data.s.body!=NULL)
321    ssiWriteString(d,p->data.s.body);
322  else
323    ssiWriteString(d,"");
324}
325
326void ssiWriteList(si_link l,lists dd)
327{
328  ssiInfo *d=(ssiInfo*)l->data;
329  int Ll=lSize(dd);
330  fprintf(d->f_write,"%d ",Ll+1);
331  int i;
332  for(i=0;i<=Ll;i++)
333  {
334    ssiWrite(l,&(dd->m[i]));
335  }
336}
337void ssiWriteIntvec(ssiInfo *d,intvec * v)
338{
339  fprintf(d->f_write,"%d ",v->length());
340  int i;
341  for(i=0;i<v->length();i++)
342  {
343    fprintf(d->f_write,"%d ",(*v)[i]);
344  }
345}
346void ssiWriteIntmat(ssiInfo *d,intvec * v)
347{
348  fprintf(d->f_write,"%d %d ",v->rows(),v->cols());
349  int i;
350  for(i=0;i<v->length();i++)
351  {
352    fprintf(d->f_write,"%d ",(*v)[i]);
353  }
354}
355
356void ssiWriteBigintmat(ssiInfo *d,bigintmat * v)
357{
358  fprintf(d->f_write,"%d %d ",v->rows(),v->cols());
359  int i;
360  for(i=0;i<v->length();i++)
361  {
362    ssiWriteBigInt(d,(*v)[i]);
363  }
364}
365
366char *ssiReadString(ssiInfo *d)
367{
368  char *buf;
369  int l;
370  l=s_readint(d->f_read);
371  buf=(char*)omAlloc0(l+1);
372  int c =s_getc(d->f_read); /* skip ' '*/
373  int ll=s_readbytes(buf,l,d->f_read);
374  //if (ll!=l) printf("want %d, got %d bytes\n",l,ll);
375  buf[l]='\0';
376  return buf;
377}
378
379int ssiReadInt(s_buff fich)
380{
381  return s_readint(fich);
382}
383
384number ssiReadBigInt(ssiInfo *d)
385{
386  int sub_type=-1;
387  sub_type=s_readint(d->f_read);
388  switch(sub_type)
389  {
390   case 3:
391     {// read int or mpz_t or mpz_t, mpz_t
392       number n=nlRInit(0);
393       s_readmpz(d->f_read,n->z);
394       n->s=sub_type;
395       return n;
396     }
397   case 4:
398     {
399       int dd;
400       dd=s_readint(d->f_read);
401       return INT_TO_SR(dd);
402     }
403   default:
404       Werror("error in reading bigint: invalid subtype %d",sub_type);
405       return NULL;
406   }
407}
408
409static number ssiReadQNumber(ssiInfo *d)
410{
411  int sub_type=-1;
412  sub_type=s_readint(d->f_read);
413  switch(sub_type)
414  {
415     case 0:
416     case 1:
417       {// read mpz_t, mpz_t
418         number n=nlRInit(0);
419         mpz_init(n->n);
420         s_readmpz(d->f_read,n->z);
421         s_readmpz(d->f_read,n->n);
422         n->s=sub_type;
423         return n;
424       }
425
426     case 3:
427       {// read mpz_t
428         number n=nlRInit(0);
429         s_readmpz(d->f_read,n->z);
430         n->s=3; /*sub_type*/
431         return n;
432       }
433     case 4:
434       {
435         LONG dd=s_readlong(d->f_read);
436         //#if SIZEOF_LONG == 8
437         return INT_TO_SR(dd);
438         //#else
439         //return nlInit(dd,NULL);
440         //#endif
441       }
442     case 5:
443     case 6:
444       {// read raw mpz_t, mpz_t
445         number n=nlRInit(0);
446         mpz_init(n->n);
447         s_readmpz_base (d->f_read,n->z, 32);
448         s_readmpz_base (d->f_read,n->n, 32);
449         n->s=sub_type-5;
450         return n;
451       }
452     case 8:
453       {// read raw mpz_t
454         number n=nlRInit(0);
455         s_readmpz_base (d->f_read,n->z, 32);
456         n->s=sub_type=3; /*subtype-5*/
457         return n;
458       }
459
460     default: Werror("error in reading number: invalid subtype %d",sub_type);
461              return NULL;
462  }
463  return NULL;
464}
465number ssiReadNumber(ssiInfo *d)
466{
467  if (rField_is_Q(d->r))
468  {
469     return ssiReadQNumber(d);
470  }
471  else if (rField_is_Zp(d->r))
472  {
473    // read int
474    int dd;
475    dd=s_readint(d->f_read);
476    return (number)(long)dd;
477  }
478  else Werror("coeffs not implemented");
479  return NULL;
480}
481
482ring ssiReadRing(ssiInfo *d)
483{
484/* syntax is <ch> <N> <l1> <v1> ...<lN> <vN> <number of orderings> <ord1> <block0_1> <block1_1> .... */
485  int ch, N,i,l;
486  char **names;
487  ch=s_readint(d->f_read);
488  N=s_readint(d->f_read);
489  if (N!=0)
490  {
491    names=(char**)omAlloc(N*sizeof(char*));
492    for(i=0;i<N;i++)
493    {
494      names[i]=ssiReadString(d);
495    }
496  }
497  // read the orderings:
498  int num_ord; // number of orderings
499  num_ord=s_readint(d->f_read);
500  int *ord=(int *)omAlloc0((num_ord+1)*sizeof(int));
501  int *block0=(int *)omAlloc0((num_ord+1)*sizeof(int));
502  int *block1=(int *)omAlloc0((num_ord+1)*sizeof(int));
503  int **wvhdl=(int**)omAlloc0((num_ord+1)*sizeof(int*));
504  for(i=0;i<num_ord;i++)
505  {
506    ord[i]=s_readint(d->f_read);
507    block0[i]=s_readint(d->f_read);
508    block1[i]=s_readint(d->f_read);
509    switch(ord[i])
510    {
511      case ringorder_a:
512      case ringorder_wp:
513      case ringorder_Wp:
514      case ringorder_ws:
515      case ringorder_Ws:
516      case ringorder_aa:
517      {
518        wvhdl[i]=(int*)omAlloc((block1[i]-block0[i]+1)*sizeof(int));
519        int ii;
520        for(ii=block0[i];ii<=block1[i];ii++)
521          wvhdl[i][ii-block0[i]]=s_readint(d->f_read);
522      }
523      break;
524
525      case ringorder_a64:
526      case ringorder_M:
527      case ringorder_L:
528      case ringorder_IS:
529        Werror("ring oder not implemented for ssi:%d",ord[i]);
530        break;
531
532      default: break;
533    }
534  }
535  if (N==0)
536  {
537    omFree(ord);
538    omFree(block0);
539    omFree(block1);
540    omFree(wvhdl);
541    return NULL;
542  }
543  else return rDefault(ch,N,names,num_ord,ord,block0,block1,wvhdl);
544
545  return rDefault(ch,N,names,num_ord,ord,block0,block1,wvhdl);
546}
547
548poly ssiReadPoly(ssiInfo *D)
549{
550// < # of terms> < term1> < .....
551  int n,i,l;
552  n=ssiReadInt(D->f_read);
553  //Print("poly: terms:%d\n",n);
554  poly p;
555  int j;
556  j=0;
557  poly ret=NULL;
558  poly prev=NULL;
559  for(l=0;l<n;l++) // read n terms
560  {
561// coef,comp.exp1,..exp N
562    p=p_Init(D->r);
563    pGetCoeff(p)=ssiReadNumber(D);
564    int d;
565    d=s_readint(D->f_read);
566    p_SetComp(p,d,D->r);
567    for(i=1;i<=rVar(D->r);i++)
568    {
569      d=s_readint(D->f_read);
570      p_SetExp(p,i,d,D->r);
571    }
572    p_Setm(p,D->r);
573    p_Test(p,D->r);
574    if (ret==NULL) ret=p;
575    else           pNext(prev)=p;
576    prev=p;
577 }
578 return ret;
579}
580
581ideal ssiReadIdeal(ssiInfo *d)
582{
583  int n,i;
584  ideal I;
585  n=s_readint(d->f_read);
586  I=idInit(n,1);
587  for(i=0;i<IDELEMS(I);i++) // read n terms
588  {
589    I->m [i]=ssiReadPoly(d);
590  }
591  return I;
592}
593
594matrix ssiReadMatrix(ssiInfo *d)
595{
596  int n,m,i,j;
597  m=s_readint(d->f_read);
598  n=s_readint(d->f_read);
599  matrix M=mpNew(m,n);
600  poly p;
601  for(int i=1;i<=MATROWS(M);i++)
602    for(int j=1;j<=MATCOLS(M);j++)
603    {
604      p=ssiReadPoly(d);
605      MATELEM(M,i,j)=p;
606    }
607  return M;
608}
609
610command ssiReadCommand(si_link l)
611{
612  ssiInfo *d=(ssiInfo*)l->data;
613  // syntax: <num ops> <operation> <op1> <op2> ....
614  command D=(command)omAlloc0(sizeof(*D));
615  int argc,op;
616  argc=s_readint(d->f_read);
617  op=s_readint(d->f_read);
618  D->argc=argc; D->op=op;
619  leftv v;
620  if (argc >0)
621  {
622    v=ssiRead1(l);
623    memcpy(&(D->arg1),v,sizeof(*v));
624    omFreeBin(v,sleftv_bin);
625  }
626  if (argc <4)
627  {
628    if (D->argc >1)
629    {
630      v=ssiRead1(l);
631      memcpy(&(D->arg2),v,sizeof(*v));
632      omFreeBin(v,sleftv_bin);
633    }
634    if (D->argc >2)
635    {
636      v=ssiRead1(l);
637      memcpy(&(D->arg3),v,sizeof(*v));
638      omFreeBin(v,sleftv_bin);
639    }
640  }
641  else
642  {
643    leftv prev=&(D->arg1);
644    argc--;
645    while(argc >0)
646    {
647      v=ssiRead1(l);
648      prev->next=v;
649      prev=v;
650      argc--;
651    }
652  }
653  return D;
654}
655
656procinfov ssiReadProc(ssiInfo *d)
657{
658  char *s=ssiReadString(d);
659  procinfov p=(procinfov)omAlloc0Bin(procinfo_bin);
660  p->language=LANG_SINGULAR;
661  p->libname=omStrDup("");
662  p->procname=omStrDup("");
663  p->data.s.body=s;
664  return p;
665}
666lists ssiReadList(si_link l)
667{
668  ssiInfo *d=(ssiInfo*)l->data;
669  int nr;
670  nr=s_readint(d->f_read);
671  lists L=(lists)omAlloc(sizeof(*L));
672  L->Init(nr);
673
674  int i;
675  leftv v;
676  for(i=0;i<nr;i++)
677  {
678    v=ssiRead1(l);
679    memcpy(&(L->m[i]),v,sizeof(*v));
680    omFreeBin(v,sleftv_bin);
681  }
682  return L;
683}
684intvec* ssiReadIntvec(ssiInfo *d)
685{
686  int nr;
687  nr=s_readint(d->f_read);
688  intvec *v=new intvec(nr);
689  for(int i=0;i<nr;i++)
690  {
691    (*v)[i]=s_readint(d->f_read);
692  }
693  return v;
694}
695intvec* ssiReadIntmat(ssiInfo *d)
696{
697  int r,c;
698  r=s_readint(d->f_read);
699  c=s_readint(d->f_read);
700  intvec *v=new intvec(r,c,0);
701  for(int i=0;i<r*c;i++)
702  {
703    (*v)[i]=s_readint(d->f_read);
704  }
705  return v;
706}
707bigintmat* ssiReadBigintmat(ssiInfo *d)
708{
709  int r,c;
710  r=s_readint(d->f_read);
711  c=s_readint(d->f_read);
712  bigintmat *v=new bigintmat(r,c,coeffs_BIGINT);
713  for(int i=0;i<r*c;i++)
714  {
715    (*v)[i]=ssiReadBigInt(d);
716  }
717  return v;
718}
719
720void ssiReadBlackbox(leftv res, si_link l)
721{
722  ssiInfo *d=(ssiInfo*)l->data;
723  int throwaway;
724  throwaway=s_readint(d->f_read);
725  char *name=ssiReadString(d);
726  int tok;
727  blackboxIsCmd(name,tok);
728  if (tok>MAX_TOK)
729  {
730    blackbox *b=getBlackboxStuff(tok);
731    res->rtyp=tok;
732    b->blackbox_deserialize(&b,&(res->data),l);
733  }
734  else
735  {
736    Werror("blackbox %s not found",name);
737  }
738}
739
740//**************************************************************************/
741
742BOOLEAN ssiOpen(si_link l, short flag, leftv u)
743{
744  if (l!=NULL)
745  {
746    const char *mode;
747    ssiInfo *d=(ssiInfo*)omAlloc0(sizeof(ssiInfo));
748    if (flag & SI_LINK_OPEN)
749    {
750      if (l->mode[0] != '\0' && (strcmp(l->mode, "r") == 0))
751        flag = SI_LINK_READ;
752      else flag = SI_LINK_WRITE;
753    }
754
755    if (flag == SI_LINK_READ) mode = "r";
756    else if (strcmp(l->mode, "w") == 0) mode = "w";
757    else if (strcmp(l->mode, "fork") == 0) mode = "fork";
758    else if (strcmp(l->mode, "tcp") == 0) mode = "tcp";
759    else if (strcmp(l->mode, "connect") == 0) mode = "connect";
760    else mode = "a";
761
762
763    SI_LINK_SET_OPEN_P(l, flag);
764    l->data=d;
765    omFree(l->mode);
766    l->mode = omStrDup(mode);
767
768    if (l->name[0] == '\0')
769    {
770      if (strcmp(mode,"fork")==0)
771      {
772        link_list n=(link_list)omAlloc(sizeof(link_struct));
773        n->u=u;
774        n->l=l;
775        n->next=(void *)ssiToBeClosed;
776        ssiToBeClosed=n;
777
778        int pc[2];
779        int cp[2];
780        pipe(pc);
781        pipe(cp);
782        pid_t pid = fork();
783        if (pid == -1 && errno == EAGAIN)   // RLIMIT_NPROC too low?
784        {
785          raise_rlimit_nproc();
786          pid = fork();
787        }
788        if (pid == -1)
789        {
790          WerrorS("could not fork");
791        }
792        if (pid==0) /*fork: child*/
793        {
794          /* block SIGINT */
795          sigset_t sigint;
796          sigemptyset(&sigint);
797          sigaddset(&sigint, SIGINT);
798          sigprocmask(SIG_BLOCK, &sigint, NULL);
799
800          link_list hh=(link_list)ssiToBeClosed->next;
801          /* we know: l is the first entry in ssiToBeClosed-list */
802          while(hh!=NULL)
803          {
804            SI_LINK_SET_CLOSE_P(hh->l);
805            ssiInfo *dd=(ssiInfo*)hh->l->data;
806            s_close(dd->f_read);
807            s_free(dd->f_read);
808            fclose(dd->f_write);
809            if (dd->r!=NULL) rKill(dd->r);
810            omFreeSize((ADDRESS)dd,(sizeof *dd));
811            hh->l->data=NULL;
812            link_list nn=(link_list)hh->next;
813            omFree(hh);
814            hh=nn;
815          }
816          ssiToBeClosed->next=NULL;
817#ifdef HAVE_SIMPLEIPC
818          memset(sem_acquired, 0, SIPC_MAX_SEMAPHORES*sizeof(sem_acquired[0]));
819#endif   // HAVE_SIMPLEIPC
820          si_close(pc[1]); si_close(cp[0]);
821          d->f_write=fdopen(cp[1],"w");
822          d->f_read=s_open(pc[0]);
823          d->fd_read=pc[0];
824          d->fd_write=cp[1];
825          l->data=d;
826          omFree(l->mode);
827          l->mode = omStrDup(mode);
828          singular_in_batchmode=TRUE;
829          SI_LINK_SET_RW_OPEN_P(l);
830          //myynest=0;
831          fe_fgets_stdin=fe_fgets_dummy;
832          if ((u!=NULL)&&(u->rtyp==IDHDL))
833          {
834            idhdl h=(idhdl)u->data;
835            h->lev=0;
836          }
837          loop
838          {
839            leftv h=ssiRead1(l); /*contains an exit.... */
840            if (feErrors != NULL && *feErrors != '\0')
841            {
842              // handle errors:
843              PrintS(feErrors); /* currently quite simple */
844              *feErrors = '\0';
845            }
846            ssiWrite(l,h);
847            h->CleanUp();
848            omFreeBin(h, sleftv_bin);
849          }
850          /* never reached*/
851        }
852        else if (pid>0) /*fork: parent*/
853        {
854          d->pid=pid;
855          si_close(pc[0]); si_close(cp[1]);
856          d->f_write=fdopen(pc[1],"w");
857          d->f_read=s_open(cp[0]);
858          d->fd_read=cp[0];
859          d->fd_write=pc[1];
860          SI_LINK_SET_RW_OPEN_P(l);
861          d->send_quit_at_exit=1;
862        }
863        else
864        {
865          Werror("fork failed (%d)",errno);
866          l->data=NULL;
867          omFree(d);
868          return TRUE;
869        }
870      }
871      // ---------------------------------------------------------------------
872      else if (strcmp(mode,"tcp")==0)
873      {
874        int sockfd, newsockfd, portno, clilen;
875        struct sockaddr_in serv_addr, cli_addr;
876        int n;
877        sockfd = socket(AF_INET, SOCK_STREAM, 0);
878        if(sockfd < 0)
879        {
880          WerrorS("ERROR opening socket");
881          l->data=NULL;
882          omFree(d);
883          return TRUE;
884        }
885        memset((char *) &serv_addr,0, sizeof(serv_addr));
886        portno = 1025;
887        serv_addr.sin_family = AF_INET;
888        serv_addr.sin_addr.s_addr = INADDR_ANY;
889        do
890        {
891          portno++;
892          serv_addr.sin_port = htons(portno);
893          if(portno > 50000)
894          {
895            WerrorS("ERROR on binding (no free port available?)");
896            l->data=NULL;
897            omFree(d);
898            return TRUE;
899          }
900        }
901        while(bind(sockfd, (struct sockaddr *) &serv_addr, sizeof(serv_addr)) < 0);
902        Print("waiting on port %d\n", portno);mflush();
903        listen(sockfd,1);
904        newsockfd = si_accept(sockfd, (struct sockaddr *) &cli_addr, (socklen_t *)&clilen);
905        if(newsockfd < 0)
906        {
907          WerrorS("ERROR on accept");
908          l->data=NULL;
909          omFree(d);
910          return TRUE;
911        }
912        PrintS("client accepted\n");
913        d->fd_read = newsockfd;
914        d->fd_write = newsockfd;
915        d->f_read = s_open(newsockfd);
916        d->f_write = fdopen(newsockfd, "w");
917        SI_LINK_SET_RW_OPEN_P(l);
918        si_close(sockfd);
919      }
920      // no ssi-Link on stdin or stdout
921      else
922      {
923        Werror("invalid mode >>%s<< for ssi",mode);
924        l->data=NULL;
925        omFree(d);
926        return TRUE;
927      }
928    }
929    // =========================================================================
930    else /*l->name=NULL*/
931    {
932      // tcp mode
933      if(strcmp(mode,"tcp")==0)
934      {
935        int sockfd, newsockfd, portno, clilen;
936        struct sockaddr_in serv_addr, cli_addr;
937        int n;
938        sockfd = socket(AF_INET, SOCK_STREAM, 0);
939        if(sockfd < 0)
940        {
941          WerrorS("ERROR opening socket");
942          l->data=NULL;
943          omFree(d);
944          return TRUE;
945        }
946        memset((char *) &serv_addr,0, sizeof(serv_addr));
947        portno = 1025;
948        serv_addr.sin_family = AF_INET;
949        serv_addr.sin_addr.s_addr = INADDR_ANY;
950        do
951        {
952          portno++;
953          serv_addr.sin_port = htons(portno);
954          if(portno > 50000)
955          {
956            WerrorS("ERROR on binding (no free port available?)");
957            l->data=NULL;
958            return TRUE;
959          }
960        }
961        while(bind(sockfd, (struct sockaddr *) &serv_addr, sizeof(serv_addr)) < 0);
962        //Print("waiting on port %d\n", portno);mflush();
963        listen(sockfd,1);
964        char* cli_host = (char*)omAlloc(256);
965        char* path = (char*)omAlloc(1024);
966        int r = si_sscanf(l->name,"%255[^:]:%s",cli_host,path);
967        if(r == 0)
968        {
969          WerrorS("ERROR: no host specified");
970          l->data=NULL;
971          omFree(d);
972          omFree(path);
973          omFree(cli_host);
974          return TRUE;
975        }
976        else if(r == 1)
977        {
978          WarnS("program not specified, using /usr/local/bin/Singular");
979          strcpy(path,"/usr/local/bin/Singular");
980        }
981        char* ssh_command = (char*)omAlloc(256);
982        char* ser_host = (char*)omAlloc(64);
983        gethostname(ser_host,64);
984        sprintf(ssh_command,"ssh %s %s -q --batch --link=ssi --MPhost=%s --MPport=%d &",cli_host,path,ser_host,portno);
985        //Print("client on %s started:%s\n",cli_host,path);
986        omFree(path);
987        omFree(cli_host);
988        if (TEST_OPT_PROT) { Print("running >>%s<<\n",ssh_command); }
989        system(ssh_command);
990        omFree(ssh_command);
991        omFree(ser_host);
992        clilen = sizeof(cli_addr);
993        newsockfd = si_accept(sockfd, (struct sockaddr *) &cli_addr, (socklen_t *)&clilen);
994        if(newsockfd < 0)
995        {
996          WerrorS("ERROR on accept");
997          l->data=NULL;
998          omFree(d);
999          return TRUE;
1000        }
1001        //PrintS("client accepted\n");
1002        d->fd_read = newsockfd;
1003        d->fd_write = newsockfd;
1004        d->f_read = s_open(newsockfd);
1005        d->f_write = fdopen(newsockfd, "w");
1006        si_close(sockfd);
1007        SI_LINK_SET_RW_OPEN_P(l);
1008        d->send_quit_at_exit=1;
1009        link_list newlink=(link_list)omAlloc(sizeof(link_struct));
1010        newlink->u=u;
1011        newlink->l=l;
1012        newlink->next=(void *)ssiToBeClosed;
1013        ssiToBeClosed=newlink;
1014        fprintf(d->f_write,"98 %d %d %u %u\n",SSI_VERSION,MAX_TOK,si_opt_1,si_opt_2);
1015      }
1016      // ----------------------------------------------------------------------
1017      else if(strcmp(mode,"connect")==0)
1018      {
1019        char* host = (char*)omAlloc(256);
1020        int sockfd, portno, n;
1021        struct sockaddr_in serv_addr;
1022        struct hostent *server;
1023
1024        si_sscanf(l->name,"%255[^:]:%d",host,&portno);
1025        //Print("connect to host %s, port %d\n",host,portno);mflush();
1026        if (portno!=0)
1027        {
1028          sockfd = socket(AF_INET, SOCK_STREAM, 0);
1029          if (sockfd < 0) { WerrorS("ERROR opening socket"); return TRUE; }
1030          server = gethostbyname(host);
1031          if (server == NULL) {  WerrorS("ERROR, no such host");  return TRUE; }
1032          memset((char *) &serv_addr, 0, sizeof(serv_addr));
1033          serv_addr.sin_family = AF_INET;
1034          memcpy((char *)&serv_addr.sin_addr.s_addr,
1035                (char *)server->h_addr,
1036                server->h_length);
1037          serv_addr.sin_port = htons(portno);
1038          if (si_connect(sockfd,(sockaddr*)&serv_addr,sizeof(serv_addr)) < 0)
1039          { Werror("ERROR connecting(errno=%d)",errno); return TRUE; }
1040          //PrintS("connected\n");mflush();
1041          d->f_read=s_open(sockfd);
1042          d->fd_read=sockfd;
1043          d->f_write=fdopen(sockfd,"w");
1044          d->fd_write=sockfd;
1045          SI_LINK_SET_RW_OPEN_P(l);
1046          omFree(host);
1047        }
1048        else
1049        {
1050          l->data=NULL;
1051          omFree(d);
1052          return TRUE;
1053        }
1054      }
1055      // ======================================================================
1056      else
1057      {
1058        // normal link to a file
1059        FILE *outfile;
1060        char *filename=l->name;
1061
1062        if(filename[0]=='>')
1063        {
1064          if (filename[1]=='>')
1065          {
1066            filename+=2;
1067            mode = "a";
1068          }
1069          else
1070          {
1071            filename++;
1072            mode="w";
1073          }
1074        }
1075        outfile=myfopen(filename,mode);
1076        if (outfile!=NULL)
1077        {
1078          if (strcmp(l->mode,"r")==0)
1079          {
1080            fclose(outfile);
1081            d->f_read=s_open_by_name(filename);
1082          }
1083          else
1084          {
1085            d->f_write = outfile;
1086            fprintf(d->f_write,"98 %d %d %u %u\n",SSI_VERSION,MAX_TOK,si_opt_1,si_opt_2);
1087          }
1088        }
1089        else
1090        {
1091          omFree(d);
1092          l->data=NULL;
1093          return TRUE;
1094        }
1095      }
1096    }
1097  }
1098
1099  return FALSE;
1100}
1101
1102//**************************************************************************/
1103BOOLEAN ssiPrepClose(si_link l)
1104{
1105  if (l!=NULL)
1106  {
1107    ssiInfo *d = (ssiInfo *)l->data;
1108    if (d!=NULL)
1109    {
1110      if (d->send_quit_at_exit)
1111      {
1112        fputs("99\n",d->f_write);
1113        fflush(d->f_write);
1114      }
1115      d->quit_sent=1;
1116    }
1117  }
1118  return FALSE;
1119}
1120
1121BOOLEAN ssiClose(si_link l)
1122{
1123  if (l!=NULL)
1124  {
1125    SI_LINK_SET_CLOSE_P(l);
1126    ssiInfo *d = (ssiInfo *)l->data;
1127    if (d!=NULL)
1128    {
1129      if ((d->send_quit_at_exit)
1130      && (d->quit_sent==0))
1131      {
1132        fputs("99\n",d->f_write);
1133        fflush(d->f_write);
1134      }
1135      if (d->r!=NULL) rKill(d->r);
1136      if ((d->pid!=0)
1137      && (si_waitpid(d->pid,NULL,WNOHANG)==0))
1138      {
1139        struct timespec t;
1140        t.tv_sec=0;
1141        t.tv_nsec=100000000; // <=100 ms
1142        struct timespec rem;
1143        int r;
1144        int wait=0;
1145        do
1146        {
1147          r = nanosleep(&t, &rem);
1148          t = rem;
1149        } while ((r < 0) && (errno == EINTR)
1150            && ((wait = si_waitpid(d->pid,NULL,WNOHANG)) == 0));
1151        if ((r == 0) && (wait == 0))
1152        {
1153          kill(d->pid,15);
1154          t.tv_sec=5; // <=5s
1155          t.tv_nsec=0;
1156          do
1157          {
1158            r = nanosleep(&t, &rem);
1159            t = rem;
1160          } while ((r < 0) && (errno == EINTR)
1161              && ((wait = si_waitpid(d->pid,NULL,WNOHANG)) == 0));
1162          if ((r == 0) && (wait == 0))
1163          {
1164            kill(d->pid,9); // just to be sure
1165            si_waitpid(d->pid,NULL,0);
1166          }
1167        }
1168      }
1169      if (d->f_read!=NULL) s_close(d->f_read);
1170      if (d->f_read!=NULL) s_free(d->f_read);
1171      if (d->f_write!=NULL) fclose(d->f_write);
1172      if ((strcmp(l->mode,"tcp")==0)
1173      || (strcmp(l->mode,"fork")==0))
1174      {
1175        link_list hh=ssiToBeClosed;
1176        if (hh!=NULL)
1177        {
1178          if (hh->l==l)
1179          {
1180             ssiToBeClosed=(link_list)hh->next;
1181             omFreeSize(hh,sizeof(link_struct));
1182          }
1183          else while(hh->next!=NULL)
1184          {
1185            link_list hhh=(link_list)hh->next;
1186            if (hhh->l==l)
1187            {
1188              hh->next=hhh->next;
1189              omFreeSize(hhh,sizeof(link_struct));
1190              break;
1191            }
1192            else
1193              hh=(link_list)hh->next;
1194          }
1195        }
1196      }
1197      omFreeSize((ADDRESS)d,(sizeof *d));
1198    }
1199    l->data=NULL;
1200  }
1201  return FALSE;
1202}
1203
1204//**************************************************************************/
1205leftv ssiRead1(si_link l)
1206{
1207  ssiInfo *d = (ssiInfo *)l->data;
1208  leftv res=(leftv)omAlloc0(sizeof(sleftv));
1209  int t=0;
1210  t=s_readint(d->f_read);
1211  //Print("got type %d\n",t);
1212  switch(t)
1213  {
1214    case 1:res->rtyp=INT_CMD;
1215           res->data=(char *)(long)ssiReadInt(d->f_read);
1216           break;
1217    case 2:res->rtyp=STRING_CMD;
1218           res->data=(char *)ssiReadString(d);
1219           break;
1220    case 3:res->rtyp=NUMBER_CMD;
1221           res->data=(char *)ssiReadNumber(d);
1222           break;
1223    case 4:res->rtyp=BIGINT_CMD;
1224           res->data=(char *)ssiReadBigInt(d);
1225           break;
1226    case 15:
1227    case 5:{
1228             d->r=ssiReadRing(d);
1229             res->rtyp=RING_CMD;
1230             res->data=(char*)d->r;
1231             // we are in the top-level, so set the basering to d->r:
1232             if (d->r!=NULL)
1233             {
1234               d->r->ref++;
1235               ssiSetCurrRing(d->r);
1236             }
1237             if (t==15) return ssiRead1(l);
1238           }
1239           break;
1240    case 6:res->rtyp=POLY_CMD;
1241           if (d->r==NULL) goto no_ring;
1242           res->data=(char*)ssiReadPoly(d);
1243           break;
1244    case 7:res->rtyp=IDEAL_CMD;
1245           if (d->r==NULL) goto no_ring;
1246           res->data=(char*)ssiReadIdeal(d);
1247           break;
1248    case 8:res->rtyp=MATRIX_CMD;
1249           if (d->r==NULL) goto no_ring;
1250           res->data=(char*)ssiReadMatrix(d);
1251           break;
1252    case 9:res->rtyp=VECTOR_CMD;
1253           if (d->r==NULL) goto no_ring;
1254           res->data=(char*)ssiReadPoly(d);
1255           break;
1256    case 10:res->rtyp=MODUL_CMD;
1257           if (d->r==NULL) goto no_ring;
1258           res->data=(char*)ssiReadIdeal(d);
1259           break;
1260    case 11:
1261           {
1262             res->rtyp=COMMAND;
1263             res->data=ssiReadCommand(l);
1264             int nok=res->Eval();
1265             if (nok) WerrorS("error in eval");
1266             break;
1267           }
1268    case 12: /*DEF_CMD*/
1269           {
1270             res->rtyp=0;
1271             res->name=(char *)ssiReadString(d);
1272             int nok=res->Eval();
1273             if (nok) WerrorS("error in name lookup");
1274             break;
1275           }
1276    case 13: res->rtyp=PROC_CMD;
1277             res->data=ssiReadProc(d);
1278             break;
1279    case 14: res->rtyp=LIST_CMD;
1280             res->data=ssiReadList(l);
1281             break;
1282    case 16: res->rtyp=NONE; res->data=NULL;
1283             break;
1284    case 17: res->rtyp=INTVEC_CMD;
1285             res->data=ssiReadIntvec(d);
1286             break;
1287    case 18: res->rtyp=INTMAT_CMD;
1288             res->data=ssiReadIntmat(d);
1289             break;
1290    case 19: res->rtyp=BIGINTMAT_CMD;
1291             res->data=ssiReadBigintmat(d);
1292             break;
1293    case 20: ssiReadBlackbox(res,l);
1294             break;
1295    // ------------
1296    case 98: // version
1297             {
1298                int n98_v,n98_m;
1299                BITSET n98_o1,n98_o2;
1300                n98_v=s_readint(d->f_read);
1301                n98_m=s_readint(d->f_read);
1302                n98_o1=s_readint(d->f_read);
1303                n98_o2=s_readint(d->f_read);
1304                if ((n98_v!=SSI_VERSION) ||(n98_m!=MAX_TOK))
1305                {
1306                  Print("incompatible versions of ssi: %d/%d vs %d/%d",
1307                                  SSI_VERSION,MAX_TOK,n98_v,n98_m);
1308                }
1309                #ifndef NDEBUG
1310                if (TEST_OPT_DEBUG)
1311                  Print("// opening ssi-%d, MAX_TOK=%d\n",n98_v,n98_m);
1312                #endif
1313                si_opt_1=n98_o1;
1314                si_opt_2=n98_o2;
1315                return ssiRead1(l);
1316             }
1317    case 99: ssiClose(l); m2_end(0);
1318    case 0: if (s_iseof(d->f_read))
1319            {
1320              ssiClose(l);
1321              res->rtyp=DEF_CMD;
1322              break;
1323            }
1324    default: Werror("not implemented (t:%d)",t);
1325             omFreeSize(res,sizeof(sleftv));
1326             res=NULL;
1327             break;
1328  }
1329  return res;
1330no_ring: WerrorS("no ring");
1331  omFreeSize(res,sizeof(sleftv));
1332  return NULL;
1333}
1334//**************************************************************************/
1335BOOLEAN ssiWrite(si_link l, leftv data)
1336{
1337  if(SI_LINK_W_OPEN_P(l)==0)
1338     if (slOpen(l,SI_LINK_OPEN|SI_LINK_WRITE,NULL)) return TRUE;
1339  ssiInfo *d = (ssiInfo *)l->data;
1340  d->level++;
1341  //FILE *fich=d->f;
1342  while (data!=NULL)
1343  {
1344    int tt=data->Typ();
1345    void *dd=data->Data();
1346    if ((dd==NULL) && (data->name!=NULL) && (tt==0)) tt=DEF_CMD;
1347      // return pure undefined names as def
1348
1349    switch(tt /*data->Typ()*/)
1350    {
1351          case NONE/* nothing*/:fputs("16 ",d->f_write);
1352                          break;
1353          case STRING_CMD: fputs("2 ",d->f_write);
1354                           ssiWriteString(d,(char *)dd);
1355                           break;
1356          case INT_CMD: fputs("1 ",d->f_write);
1357                        ssiWriteInt(d,(int)(long)dd);
1358                        break;
1359          case BIGINT_CMD:fputs("4 ",d->f_write);
1360                        ssiWriteBigInt(d,(number)dd);
1361                        break;
1362          case NUMBER_CMD:
1363                          if (d->r!=currRing)
1364                          {
1365                            fputs("15 ",d->f_write);
1366                            ssiWriteRing(d,currRing);
1367                            if (d->level<=1) fputc('\n',d->f_write);
1368                          }
1369                          fputs("3 ",d->f_write);
1370                          ssiWriteNumber(d,(number)dd);
1371                        break;
1372          case RING_CMD:fputs("5 ",d->f_write);
1373                        ssiWriteRing(d,(ring)dd);
1374                        break;
1375          case POLY_CMD:
1376          case VECTOR_CMD:
1377                        if (d->r!=currRing)
1378                        {
1379                          fputs("15 ",d->f_write);
1380                          ssiWriteRing(d,currRing);
1381                          if (d->level<=1) fputc('\n',d->f_write);
1382                        }
1383                        if(tt==POLY_CMD) fputs("6 ",d->f_write);
1384                        else             fputs("9 ",d->f_write);
1385                        ssiWritePoly(d,tt,(poly)dd);
1386                        break;
1387          case IDEAL_CMD:
1388          case MODUL_CMD:
1389          case MATRIX_CMD:
1390                        if (d->r!=currRing)
1391                        {
1392                          fputs("15 ",d->f_write);
1393                          ssiWriteRing(d,currRing);
1394                          if (d->level<=1) fputc('\n',d->f_write);
1395                        }
1396                        if(tt==IDEAL_CMD)       fputs("7 ",d->f_write);
1397                        else if(tt==MATRIX_CMD) fputs("8 ",d->f_write);
1398                        else                    fputs("10 ",d->f_write);
1399                        ssiWriteIdeal(d,tt,(ideal)dd);
1400                        break;
1401          case COMMAND:
1402                   fputs("11 ",d->f_write);
1403                   ssiWriteCommand(l,(command)dd);
1404                   break;
1405          case DEF_CMD: /* not evaluated stuff in quotes */
1406                   fputs("12 ",d->f_write);
1407                   ssiWriteString(d,data->Name());
1408                   break;
1409          case PROC_CMD:
1410                   fputs("13 ",d->f_write);
1411                   ssiWriteProc(d,(procinfov)dd);
1412                   break;
1413          case LIST_CMD:
1414                   fputs("14 ",d->f_write);
1415                   ssiWriteList(l,(lists)dd);
1416                   break;
1417          case INTVEC_CMD:
1418                   fputs("17 ",d->f_write);
1419                   ssiWriteIntvec(d,(intvec *)dd);
1420                   break;
1421          case INTMAT_CMD:
1422                   fputs("18 ",d->f_write);
1423                   ssiWriteIntmat(d,(intvec *)dd);
1424                   break;
1425          case BIGINTMAT_CMD:
1426                   fputs("19 ",d->f_write);
1427                   ssiWriteBigintmat(d,(bigintmat *)dd);
1428                   break;
1429          default:
1430            if (tt>MAX_TOK)
1431            {
1432              blackbox *b=getBlackboxStuff(tt);
1433              fputs("20 ",d->f_write);
1434              b->blackbox_serialize(b,dd,l);
1435            }
1436            else
1437            {
1438              Werror("not implemented (t:%d, rtyp:%d)",tt, data->rtyp);
1439              d->level=0;
1440              return TRUE;
1441            }
1442            break;
1443    }
1444    if (d->level<=1) { fputc('\n',d->f_write); fflush(d->f_write); }
1445    data=data->next;
1446  }
1447  d->level--;
1448  return FALSE;
1449}
1450
1451BOOLEAN ssiGetDump(si_link l);
1452BOOLEAN ssiDump(si_link l);
1453
1454si_link_extension slInitSsiExtension(si_link_extension s)
1455{
1456  s->Open=ssiOpen;
1457  s->Close=ssiClose;
1458  s->Kill=ssiClose;
1459  s->Read=ssiRead1;
1460  s->Read2=(slRead2Proc)NULL;
1461  s->Write=ssiWrite;
1462  s->Dump=ssiDump;
1463  s->GetDump=ssiGetDump;
1464
1465  s->Status=slStatusSsi;
1466  s->type="ssi";
1467  return s;
1468}
1469
1470const char* slStatusSsi(si_link l, const char* request)
1471{
1472  ssiInfo *d=(ssiInfo*)l->data;
1473  if (d==NULL) return "not open";
1474  if (((strcmp(l->mode,"fork")==0)
1475  ||(strcmp(l->mode,"tcp")==0)
1476  ||(strcmp(l->mode,"connect")==0))
1477  && (strcmp(request, "read") == 0))
1478  {
1479    fd_set  mask, fdmask;
1480    struct timeval wt;
1481    if (s_isready(d->f_read)) return "ready";
1482    loop
1483    {
1484      /* Don't block. Return socket status immediately. */
1485      wt.tv_sec  = 0;
1486      wt.tv_usec = 0;
1487
1488      FD_ZERO(&mask);
1489      FD_SET(d->fd_read, &mask);
1490      //Print("test fd %d\n",d->fd_read);
1491    /* check with select: chars waiting: no -> not ready */
1492      switch (si_select(d->fd_read+1, &mask, NULL, NULL, &wt))
1493      {
1494        case 0: /* not ready */ return "not ready";
1495        case -1: /*error*/      return "error";
1496        case 1: /*ready ? */    break;
1497      }
1498    /* yes: read 1 char*/
1499    /* if \n, check again with select else ungetc(c), ready*/
1500      int c=s_getc(d->f_read);
1501      //Print("try c=%d\n",c);
1502      if (c== -1) return "eof"; /* eof or error */
1503      else if (isdigit(c))
1504      { s_ungetc(c,d->f_read); return "ready"; }
1505      else if (c>' ')
1506      {
1507        Werror("unknown char in ssiLink(%d)",c);
1508        return "error";
1509      }
1510      /* else: next char */
1511    }
1512  }
1513  else if (strcmp(request, "read") == 0)
1514  {
1515    if (SI_LINK_R_OPEN_P(l) && (!s_iseof(d->f_read)) && (s_isready(d->f_read))) return "ready";
1516    else return "not ready";
1517  }
1518  else if (strcmp(request, "write") == 0)
1519  {
1520    if (SI_LINK_W_OPEN_P(l)) return "ready";
1521    else return "not ready";
1522  }
1523  else return "unknown status request";
1524}
1525
1526int slStatusSsiL(lists L, int timeout)
1527{
1528// input: L: a list with links of type
1529//           ssi-connect, ssi-fork, ssi-tcp, MPtcp-fork or MPtcp-launch.
1530//           Note: Not every entry in L must be set.
1531//        timeout: timeout for select in micro-seconds
1532//           or -1 for infinity
1533//           or 0 for polling
1534// returns: ERROR (via Werror): L has wrong elements or link not open
1535//           -2: select returns an error
1536//           -1: the read state of all links is eof
1537//           0:  timeout (or polling): none ready,
1538//           i>0: (at least) L[i] is ready
1539  si_link l;
1540  ssiInfo *d;
1541  int d_fd;
1542  fd_set  mask, fdmask;
1543  FD_ZERO(&fdmask);
1544  FD_ZERO(&mask);
1545  int max_fd=0; /* 1 + max fd in fd_set */
1546
1547  /* timeout */
1548  struct timeval wt;
1549  struct timeval *wt_ptr=&wt;
1550  int startingtime = getRTimer()/TIMER_RESOLUTION;  // in seconds
1551  if (timeout== -1)
1552  {
1553    wt_ptr=NULL;
1554  }
1555  else
1556  {
1557    wt.tv_sec  = timeout / 1000000;
1558    wt.tv_usec = timeout % 1000000;
1559  }
1560
1561  /* auxiliary variables */
1562  int i;
1563  int j;
1564  int k;
1565  int s;
1566  char fdmaskempty;
1567
1568  /* check the links and fill in fdmask */
1569  /* check ssi links for ungetc_buf */
1570  for(i=L->nr; i>=0; i--)
1571  {
1572    if (L->m[i].Typ()!=DEF_CMD)
1573    {
1574      if (L->m[i].Typ()!=LINK_CMD)
1575      { WerrorS("all elements must be of type link"); return -2;}
1576      l=(si_link)L->m[i].Data();
1577      if(SI_LINK_OPEN_P(l)==0)
1578      { WerrorS("all links must be open"); return -2;}
1579      if (((strcmp(l->m->type,"ssi")!=0) && (strcmp(l->m->type,"MPtcp")!=0))
1580      || ((strcmp(l->mode,"fork")!=0) && (strcmp(l->mode,"tcp")!=0)
1581        && (strcmp(l->mode,"launch")!=0) && (strcmp(l->mode,"connect")!=0)))
1582      {
1583        WerrorS("all links must be of type ssi:fork, ssi:tcp, ssi:connect");
1584        return -2;
1585      }
1586      if (strcmp(l->m->type,"ssi")==0)
1587      {
1588        d=(ssiInfo*)l->data;
1589        d_fd=d->fd_read;
1590        if (!s_isready(d->f_read))
1591        {
1592          FD_SET(d_fd, &fdmask);
1593          if (d_fd > max_fd) max_fd=d_fd;
1594        }
1595        else
1596          return i+1;
1597      }
1598      else
1599      {
1600        Werror("wrong link type >>%s<<",l->m->type);
1601        return -2;
1602      }
1603    }
1604  }
1605  max_fd++;
1606
1607do_select:
1608  /* copy fdmask to mask */
1609  FD_ZERO(&mask);
1610  for(k = 0; k < max_fd; k++)
1611  {
1612    if(FD_ISSET(k, &fdmask))
1613    {
1614      FD_SET(k, &mask);
1615    }
1616  }
1617
1618  /* check with select: chars waiting: no -> not ready */
1619  s = si_select(max_fd, &mask, NULL, NULL, wt_ptr);
1620  if (s==-1)
1621  {
1622    WerrorS("error in select call");
1623    return -2; /*error*/
1624  }
1625  if (s==0)
1626  {
1627    return 0; /*poll: not ready */
1628  }
1629  else /* s>0, at least one ready  (the number of fd which are ready is s)*/
1630  {
1631    j=0;
1632    while (j<=max_fd) { if (FD_ISSET(j,&mask)) break; j++; }
1633    for(i=L->nr; i>=0; i--)
1634    {
1635      if (L->m[i].rtyp==LINK_CMD)
1636      {
1637        l=(si_link)L->m[i].Data();
1638        if (strcmp(l->m->type,"ssi")==0)
1639        {
1640          d=(ssiInfo*)l->data;
1641          d_fd=d->fd_read;
1642          if(j==d_fd) break;
1643        }
1644        else
1645        {
1646          Werror("wrong link type >>%s<<",l->m->type);
1647          return -2;
1648        }
1649      }
1650    }
1651    // only ssi links:
1652    loop
1653    {
1654      /* yes: read 1 char*/
1655      /* if \n, check again with select else ungetc(c), ready*/
1656      /* setting: d: current ssiInfo, j current fd, i current entry in L*/
1657      int c=s_getc(d->f_read);
1658      //Print("try c=%d\n",c);
1659      if (c== -1) /* eof */
1660      {
1661        FD_CLR(j,&fdmask);
1662        fdmaskempty = 1;
1663        for(k = 0; k < max_fd; k++)
1664        {
1665          if(FD_ISSET(k, &fdmask))
1666          {
1667            fdmaskempty = 0;
1668            break;
1669          }
1670        }
1671        if(fdmaskempty)
1672        {
1673          return -1;
1674        }
1675        if(timeout != -1)
1676        {
1677          timeout = si_max(0,
1678             timeout - 1000000*(getRTimer()/TIMER_RESOLUTION - startingtime));
1679          wt.tv_sec  = timeout / 1000000;
1680          wt.tv_usec = (timeout % 1000000);
1681        }
1682        goto do_select;
1683      }
1684
1685      else if (isdigit(c))
1686      { s_ungetc(c,d->f_read); return i+1; }
1687      else if (c>' ')
1688      {
1689        Werror("unknown char in ssiLink(%d)",c);
1690        return -2;
1691      }
1692      /* else: next char */
1693    }
1694  }
1695}
1696
1697int ssiBatch(const char *host, const char * port)
1698/* return 0 on success, >0 else*/
1699{
1700  si_link l=(si_link) omAlloc0Bin(sip_link_bin);
1701  char *buf=(char*)omAlloc(256);
1702  sprintf(buf,"ssi:connect %s:%s",host,port);
1703  slInit(l, buf);
1704  if (slOpen(l,SI_LINK_OPEN,NULL)) return 1;
1705  SI_LINK_SET_RW_OPEN_P(l);
1706
1707  idhdl id = enterid(omStrDup("link_ll"), 0, LINK_CMD, &IDROOT, FALSE);
1708  IDLINK(id) = l;
1709
1710  loop
1711  {
1712    leftv h=ssiRead1(l); /*contains an exit.... */
1713    if (feErrors != NULL && *feErrors != '\0')
1714    {
1715      // handle errors:
1716      PrintS(feErrors); /* currently quite simple */
1717      *feErrors = '\0';
1718    }
1719    ssiWrite(l,h);
1720    h->CleanUp();
1721    omFreeBin(h, sleftv_bin);
1722  }
1723  /* never reached*/
1724  exit(0);
1725}
1726
1727static int ssiReserved_P=0;
1728static int ssiReserved_sockfd;
1729static  struct sockaddr_in ssiResverd_serv_addr;
1730static int  ssiReserved_Clients;
1731int ssiReservePort(int clients)
1732{
1733  if (ssiReserved_P!=0)
1734  {
1735    WerrorS("ERROR already a reverved port requested");
1736    return 0;
1737  }
1738  int portno;
1739  int n;
1740  ssiReserved_sockfd = socket(AF_INET, SOCK_STREAM, 0);
1741  if(ssiReserved_sockfd < 0)
1742  {
1743    WerrorS("ERROR opening socket");
1744    return 0;
1745  }
1746  memset((char *) &ssiResverd_serv_addr,0, sizeof(ssiResverd_serv_addr));
1747  portno = 1025;
1748  ssiResverd_serv_addr.sin_family = AF_INET;
1749  ssiResverd_serv_addr.sin_addr.s_addr = INADDR_ANY;
1750  do
1751  {
1752    portno++;
1753    ssiResverd_serv_addr.sin_port = htons(portno);
1754    if(portno > 50000)
1755    {
1756      WerrorS("ERROR on binding (no free port available?)");
1757      return 0;
1758    }
1759  }
1760  while(bind(ssiReserved_sockfd, (struct sockaddr *) &ssiResverd_serv_addr, sizeof(ssiResverd_serv_addr)) < 0);
1761  ssiReserved_P=portno;
1762  listen(ssiReserved_sockfd,clients);
1763  ssiReserved_Clients=clients;
1764  return portno;
1765}
1766
1767extern si_link_extension si_link_root;
1768si_link ssiCommandLink()
1769{
1770  if (ssiReserved_P==0)
1771  {
1772    WerrorS("ERROR no reverved port requested");
1773    return NULL;
1774  }
1775  struct sockaddr_in cli_addr;
1776  int clilen = sizeof(cli_addr);
1777  int newsockfd = si_accept(ssiReserved_sockfd, (struct sockaddr *) &cli_addr, (socklen_t *)&clilen);
1778  if(newsockfd < 0)
1779  {
1780    Werror("ERROR on accept (errno=%d)",errno);
1781    return NULL;
1782  }
1783  si_link l=(si_link) omAlloc0Bin(sip_link_bin);
1784  si_link_extension s = si_link_root;
1785  si_link_extension prev = s;
1786  while (strcmp(s->type, "ssi") != 0)
1787  {
1788    if (s->next == NULL)
1789    {
1790      prev = s;
1791      s = NULL;
1792      break;
1793    }
1794    else
1795    {
1796      s = s->next;
1797    }
1798  }
1799  if (s != NULL)
1800    l->m = s;
1801  else
1802  {
1803    si_link_extension ns = (si_link_extension)omAlloc0Bin(s_si_link_extension_bin);
1804    prev->next=slInitSsiExtension(ns);
1805    l->m = prev->next;
1806  }
1807  l->name=omStrDup("");
1808  l->mode=omStrDup("tcp");
1809  l->ref=1;
1810  ssiInfo *d=(ssiInfo*)omAlloc0(sizeof(ssiInfo));
1811  l->data=d;
1812  d->fd_read = newsockfd;
1813  d->fd_write = newsockfd;
1814  d->f_read = s_open(newsockfd);
1815  d->f_write = fdopen(newsockfd, "w");
1816  SI_LINK_SET_RW_OPEN_P(l);
1817  ssiReserved_Clients--;
1818  if (ssiReserved_Clients<=0)
1819  {
1820    ssiReserved_P=0;
1821    si_close(ssiReserved_sockfd);
1822  }
1823  return l;
1824}
1825/*---------------------------------------------------------------------*/
1826/**
1827 * @brief additional default signal handler
1828
1829  // some newer Linux version cannot have SIG_IGN for SIGCHLD,
1830  // so use this nice routine here:
1831  //  SuSe 9.x reports -1 always
1832  //  Redhat 9.x/FC x reports sometimes -1
1833  // see also: hpux_system
1834  // also needed by getrusage (timer etc.)
1835
1836 @param[in] sig
1837**/
1838/*---------------------------------------------------------------------*/
1839void sig_chld_hdl(int sig)
1840{
1841  pid_t kidpid;
1842  int status;
1843
1844  loop
1845  {
1846    kidpid = si_waitpid(-1, &status, WNOHANG);
1847    if (kidpid==-1)
1848    {
1849      /* continue on interruption (EINTR): */
1850      if (errno == EINTR) continue;
1851      /* break on anything else (EINVAL or ECHILD according to manpage): */
1852      break;
1853    }
1854    else if (kidpid==0) break; /* no more children to process, so break */
1855
1856    //printf("Child %ld terminated\n", kidpid);
1857    link_list hh=ssiToBeClosed;
1858    while((hh!=NULL)&&(ssiToBeClosed_inactive))
1859    {
1860      if((hh->l!=NULL) && (hh->l->m->Open==ssiOpen))
1861      {
1862        ssiInfo *d = (ssiInfo *)hh->l->data;
1863        if(d->pid==kidpid)
1864        {
1865          if(ssiToBeClosed_inactive)
1866          {
1867            ssiToBeClosed_inactive=FALSE;
1868            slClose(hh->l);
1869            ssiToBeClosed_inactive=TRUE;
1870            break;
1871          }
1872          else break;
1873        }
1874        else hh=(link_list)hh->next;
1875      }
1876      else hh=(link_list)hh->next;
1877    }
1878  }
1879}
1880
1881static BOOLEAN DumpSsiIdhdl(si_link l, idhdl h)
1882{
1883  int type_id = IDTYP(h);
1884
1885  // C-proc not to be dumped, also LIB-proc not
1886  if (type_id == PROC_CMD)
1887  {
1888    if (IDPROC(h)->language == LANG_C) return FALSE;
1889    if (IDPROC(h)->libname != NULL) return FALSE;
1890  }
1891  // do not dump links
1892  if (type_id == LINK_CMD) return FALSE;
1893
1894  // do not dump ssi internal rings: ssiRing*
1895  if ((type_id == RING_CMD) && (strncmp(IDID(h),"ssiRing",7)==0))
1896    return FALSE;
1897
1898  command D=(command)omAlloc0(sizeof(*D));
1899  sleftv tmp;
1900  memset(&tmp,0,sizeof(tmp));
1901  tmp.rtyp=COMMAND;
1902  tmp.data=D;
1903
1904  if (type_id == PACKAGE_CMD)
1905  {
1906    // do not dump Top
1907    if (strcmp(IDID(h), "Top") == 0) return FALSE;
1908    package p=(package)IDDATA(h);
1909    // dump Singular-packages as load("...");
1910    if (p->language==LANG_SINGULAR)
1911    {
1912      D->op=LOAD_CMD;
1913      D->argc=1;
1914      D->arg1.rtyp=STRING_CMD;
1915      D->arg1.data=p->libname;
1916      ssiWrite(l,&tmp);
1917      omFree(D);
1918      return FALSE;
1919    }
1920  }
1921
1922  // handle qrings separately
1923  //if (type_id == QRING_CMD)
1924  //  return DumpSsiQringQring(l, h);
1925
1926  // put type and name
1927  //Print("generic dump:%s,%s\n",IDID(h),Tok2Cmdname(IDTYP(h)));
1928  D->op='=';
1929  D->argc=2;
1930  D->arg1.rtyp=DEF_CMD;
1931  D->arg1.name=IDID(h);
1932  D->arg2.rtyp=IDTYP(h);
1933  D->arg2.data=IDDATA(h);
1934  ssiWrite(l,&tmp);
1935  omFree(D);
1936  return FALSE;
1937}
1938static BOOLEAN ssiDumpIter(si_link l, idhdl h)
1939{
1940  if (h == NULL) return FALSE;
1941
1942  if (ssiDumpIter(l, IDNEXT(h))) return TRUE;
1943
1944  // need to set the ring before writing it, otherwise we get in
1945  // trouble with minpoly
1946  if (IDTYP(h) == RING_CMD || IDTYP(h) == QRING_CMD)
1947    rSetHdl(h);
1948
1949  if (DumpSsiIdhdl(l, h)) return TRUE;
1950
1951  // do not dump ssi internal rings: ssiRing*
1952  // but dump objects of all other rings
1953  if ((IDTYP(h) == RING_CMD || IDTYP(h) == QRING_CMD)
1954  && (strncmp(IDID(h),"ssiRing",7)!=0))
1955    return ssiDumpIter(l, IDRING(h)->idroot);
1956  else
1957    return FALSE;
1958}
1959BOOLEAN ssiDump(si_link l)
1960{
1961  idhdl h = IDROOT, rh = currRingHdl;
1962  BOOLEAN status = ssiDumpIter(l, h);
1963
1964  //if (! status ) status = DumpAsciiMaps(fd, h, NULL);
1965
1966  if (currRingHdl != rh) rSetHdl(rh);
1967  //fprintf(fd, "option(set, intvec(%d, %d));\n", si_opt_1, si_opt_2);
1968
1969  return status;
1970}
1971BOOLEAN ssiGetDump(si_link l)
1972{
1973  ssiInfo *d=(ssiInfo*)l->data;
1974  loop
1975  {
1976    if (!SI_LINK_OPEN_P(l)) break;
1977    if (s_iseof(d->f_read)) break;
1978    leftv h=ssiRead1(l); /*contains an exit.... */
1979    if (feErrors != NULL && *feErrors != '\0')
1980    {
1981      // handle errors:
1982      PrintS(feErrors); /* currently quite simple */
1983      return TRUE;
1984      *feErrors = '\0';
1985    }
1986    h->CleanUp();
1987    omFreeBin(h, sleftv_bin);
1988  }
1989  return FALSE;
1990}
1991// ----------------------------------------------------------------
1992// format
1993// 1 int %d
1994// 2 string <len> %s
1995// 3 number
1996// 4 bigint 4 %d or 3 <mpz_t>
1997// 5 ring
1998// 6 poly
1999// 7 ideal
2000// 8 matrix
2001// 9 vector
2002// 10 module
2003// 11 command
2004// 12 def <len> %s
2005// 13 proc <len> %s
2006// 14 list %d <elem1> ....
2007// 15 setring .......
2008// 16 nothing
2009// 17 intvec <len> ...
2010// 18 intmat
2011// 19 bigintmat <r> <c> ...
2012//
2013// 20 blackbox <name> ...
2014//
2015// 98: verify version: <ssi-version> <MAX_TOK> <OPT1> <OPT2>
2016// 99: quit Singular
Note: See TracBrowser for help on using the repository browser.