/* $OpenXM: OpenXM/src/kan96xx/plugin/oxmisc.c,v 1.32 2020/10/06 11:33:47 takayama Exp $ */ #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #define SET_MYERROROUT { if (MyErrorOut == NULL) MyErrorOut=stdout; } /* It is also defined in oxmisc2.c */ FILE *MyErrorOut = NULL; /* Include files to understand object */ #include "../Kan/datatype.h" #include "../Kan/stackm.h" #include "../Kan/extern.h" #include "ox_kan.h" #include "mysig.h" void restoreLockCtrlCForOx(); // defined in Kan/stackmachine.c #define READBUFSIZE 5000 int OxVersion = 200012030; int UseOXPacketSerial = 1; int SerialOX = 1; extern int Quiet; static char *OxPortFileName = ".ox.dummy"; int readOneByte(int fd) /* blocking */ { static char data[1]; int size; int ans; int watch = 1; SET_MYERROROUT; if (fd < 0) { fprintf(MyErrorOut,"readOneByte fd < 0 ??? .\n"); return(-1); } if (oxSocketSelect0(fd,-1)) { /* block */ size = read(fd,data,1); if (size == 0) { fprintf(MyErrorOut,"oxSocketSelect0() returns 1, but there is no data. Your peer may be killed.\n"); return(-1); } return(data[0]); }else{ fprintf(MyErrorOut,"readOneByte: select error in block mode. Retrying.\n"); return(-1); } } int readOneByte_org(int fd) /* blocking */ { static char data[READBUFSIZE]; static int thisFd = -1; static int from = 0; static int to = 0; int size; int ans; int watch = 1; SET_MYERROROUT; if ((thisFd == -1) && (fd >= 0)) {thisFd = fd;} if (fd < 0) { fprintf(MyErrorOut,"readOneByte fd < 0 ??? .\n"); return(-1); } if (fd != thisFd) { fprintf(MyErrorOut,"readOneByte can be used only for one fd.\n"); fflush(NULL); return(-1); } if (to > from) { ans = data[from]; from++; if (watch) { printf("%2x ",ans); fflush(stdout); } return(ans); } while (1) { if (oxSocketSelect0(fd,-1)) { /* block */ size = read(fd,data,READBUFSIZE-1); if (size == 0) { fprintf(MyErrorOut,"oxSocketSelect0() returns 1, but there is no data. Your peer may be killed.\n"); return(-1); } from = 0; to = size; return(readOneByte(fd)); }else{ fprintf(MyErrorOut,"readOneByte: select error in block mode. Retrying.\n"); } } } int oxfdGetInt32(int fd) { char d[4]; int i; for (i=0; i<4; i++) { d[i] = readOneByte(fd); } return(ntohl(* ( (int *)d))); } int oxGetInt32(ox_stream ostream) { char d[4]; int i; for (i=0; i<4; i++) { d[i] = fp2fgetc(ostream); } return(ntohl(* ( (int *)d))); } int oxGetCMOInt32(ox_stream ostream) { int id; SET_MYERROROUT; id = oxGetInt32(ostream); if (id != CMO_INT32) { fprintf(MyErrorOut,"It is not CMO_INT32.\n"); return(0); } return(oxGetInt32(ostream)); } char *oxGetCMOString(ox_stream ostream) { int id; int size; char *r; int i; SET_MYERROROUT; id = oxGetInt32(ostream); if (id != CMO_STRING) { fprintf(MyErrorOut,"It is not CMO_STRING.\n"); return(NULL); } size = oxGetInt32(ostream); if (size <0) { return(NULL); }else{ r = (char *)mymalloc(size+1); for (i=0; iinfop)); infosize = getoaSize(mathinfo); oxSendInt32(os,CMO_MATHCAP); oxSendInt32(os,CMO_LIST); oxSendInt32(os,3); /* [0] */ oxSendInt32(os,CMO_LIST); oxSendInt32(os,infosize); oxSendCmoInt32(os,KopInteger(getoa(mathinfo,0))); for (i=1; ismSize); n = mathcap->smSize; for (i=0; ism)[i]); } /* [2] */ oxSendInt32(os,CMO_LIST); n = mathcap->oxSize; oxSendInt32(os,n); oxSendInt32(os,CMO_LIST); oxSendInt32(os,2); for (i=0; iox)[i]); /* OX_DATA_xxx tags. In case of CMO, it is CMO tags. */ oxSendInt32(os,CMO_LIST); oxSendInt32(os,mathcap->n); ncmo = mathcap->n; for (i=0; icmo)[i]); /* printf("i=%d %d, ",i,(mathcap->cmo)[i]); */ } } /* printf("\n"); fflush(stdout); */ } void oxReqMathCap(ox_stream os) { oxSendOXheader(os,OX_COMMAND,SerialOX++); oxSendInt32(os,SM_mathcap); fp2fflush(os); } void oxReqSetMathCap(ox_stream os,struct mathCap *mathcap) { oxSendOXheader(os,OX_DATA,SerialOX++); oxSendMathCap(os,mathcap); oxSendOXheader(os,OX_COMMAND,SerialOX++); oxSendInt32(os,SM_setMathCap); fp2fflush(os); } void oxReqPops(ox_stream os,int n) { oxSendOXheader(os,OX_DATA,SerialOX++); oxSendCmoInt32(os,n); oxSendOXheader(os,OX_COMMAND,SerialOX++); oxSendInt32(os,SM_pops); fp2fflush(os); } void oxReqSetName(ox_stream os,char *s) { oxSendOXheader(os,OX_DATA,SerialOX++); oxSendCmoString(os,s); oxSendOXheader(os,OX_COMMAND,SerialOX++); oxSendInt32(os,SM_setName); fp2fflush(os); } void oxReqEvalName(ox_stream os,char *s) { oxSendOXheader(os,OX_DATA,SerialOX++); oxSendCmoString(os,s); oxSendOXheader(os,OX_COMMAND,SerialOX++); oxSendInt32(os,SM_evalName); fp2fflush(os); } void oxReqExecuteStringByLocalParser(ox_stream os,char *s) { oxSendOXheader(os,OX_DATA,SerialOX++); oxSendCmoString(os,s); oxSendOXheader(os,OX_COMMAND,SerialOX++); oxSendInt32(os,SM_executeStringByLocalParser); fp2fflush(os); } void oxReqExecuteFunction(ox_stream os,char *s) { oxSendOXheader(os,OX_DATA,SerialOX++); oxSendCmoString(os,s); oxSendOXheader(os,OX_COMMAND,SerialOX++); oxSendInt32(os,SM_executeFunction); fp2fflush(os); } void oxReqExecuteFunctionWithOptionalArgument(ox_stream os,char *s) { oxSendOXheader(os,OX_DATA,SerialOX++); oxSendCmoString(os,s); oxSendOXheader(os,OX_COMMAND,SerialOX++); oxSendInt32(os,SM_executeFunctionWithOptionalArgument); fp2fflush(os); } void oxReqPopString(ox_stream os) { oxSendOXheader(os,OX_COMMAND,SerialOX++); oxSendInt32(os,SM_popString); fp2fflush(os); } void oxReqSingleOperand(ox_stream os,int smtag) { oxSendOXheader(os,OX_COMMAND,SerialOX++); oxSendInt32(os,smtag); fp2fflush(os); } void oxReqControlResetConnection(int fd) { oxfdSendOXheader(fd,OX_COMMAND,SerialOX++); oxfdSendInt32(fd,SM_control_reset_connection); fflush(NULL); } void oxReqControlKill(int fd) { oxfdSendOXheader(fd,OX_COMMAND,SerialOX++); oxfdSendInt32(fd,SM_control_kill); fflush(NULL); } void oxReqPopCMO(ox_stream os) { oxSendOXheader(os,OX_COMMAND,SerialOX++); oxSendInt32(os,SM_popCMO); fp2fflush(os); } int oxGetResultOfControlInt32(int fd) { int k; int sss; SET_MYERROROUT; k = oxfdGetOXheader(fd,&sss); if (k != OX_DATA) { fprintf(MyErrorOut,"oxGetResultOfControlInt32: wrong header."); return(-1); } k = oxfdGetInt32(fd); /* CMO_INT32 */ k = oxfdGetInt32(fd); return(k); } int oxclientMultiSelect(oxclientp clients[],int dataready[], int controlready[], int size, int t) { int i, ddd; int fd; int humanfd = 0; fd_set readfds; struct timeval timeout; SET_MYERROROUT; /** printf("(1)"); fflush(NULL); */ FD_ZERO(&readfds); timeout.tv_sec = 0; timeout.tv_usec = (long) t; ddd = 0; fd = 0; for (i=0; ihumanio) { fd = (fdcontrolport < 0) { /* For RFC_101 */ controlready[i] = 0; }else{ fd = (fdcontrolfd?clients[i]->controlfd:fd); FD_SET(clients[i]->controlfd,&readfds); if (oxSocketSelect0(clients[i]->controlfd,0)) { ddd = controlready[i] = 1; }else{ controlready[i] = 0; } } if (clients[i]->datafp2 != NULL) { fd = (fddatafp2->fd?clients[i]->datafp2->fd:fd); FD_SET(clients[i]->datafp2->fd,&readfds); if (fp2select(clients[i]->datafp2,0)) { ddd = dataready[i] = 1; }else{ dataready[i] = 0; } }else{ dataready[i] = 0; } } } if (t > 0 ) { if (ddd) return(1); if (select(fd+1,&readfds,(fd_set *)NULL,(fd_set *)NULL,&timeout)<0) { fprintf(MyErrorOut,"error"); return(-1); } return(oxclientMultiSelect(clients, dataready, controlready,size,0)); }else if (t == 0) { return(ddd); }else { /** printf("(2)"); fflush(NULL); */ if (select(fd+1,&readfds,(fd_set *)NULL,(fd_set *)NULL,(struct timeval *)NULL)<0) { fprintf(MyErrorOut,"error"); return(-1); } /** printf("(3)"); fflush(NULL); */ return(oxclientMultiSelect(clients, dataready, controlready,size,0)); } } int oxGetControl(oxclientp client) /* synchronized. */ { int ans; ox_stream os; switch (client->cstate) { case 1: ans = oxGetResultOfControlInt32(client->controlfd); client->cstate = 0; return(ans); default: fprintf(MyErrorOut,"oxGet: unknown cstate.\n"); client->cstate = -1; return(-1); } return(-1); } int oxInitClient(oxclientp client) { client->datafp2 = NULL; client->dataport = 0; client->controlport = 0; client->controlfd = 0; client->humanio = 0; client->dstate = 0; client->cstate = 0; client->id = -1; client->mathcapObjp = NULL; client->controlByteOrder = OX_BYTE_NETWORK_BYTE_ORDER; client->engineByteOrder = OX_BYTE_NETWORK_BYTE_ORDER; client->engineID = -1; return(0); } int oxIsThereErrorClient(oxclientp client) { if (client == NULL) return(1); if (client->dstate == -1) return(1); if (client->cstate == -1) return(1); return(0); } oxclientp oxCreateClient(char *sname,int portStream,int portControl, char *passControl, char *passData) /* you also need to change oxCreateClient2. */ { int v = 0; int fdControl = -1; int fdStream = -1; oxclientp client; int controlByteOrder, engineByteOrder; v = !Quiet; if (portControl != -1) { fdControl = socketConnect(sname,portControl); if (v) fprintf(stderr,"\nControl port %d : Connected.\n",portControl); } if (portStream != -1) { sleep(1); /* wait */ fdStream = socketConnect(sname,portStream); if (v) fprintf(stderr,"\nStream port %d : Connected.\n",portStream); } if (fdStream == -1 || fdControl == -1) { fprintf(stderr,"\nOpen error in oxCreateClient.\n"); return(NULL); } if (passControl != NULL) { if (v) fprintf(stderr,"Sending password %s for the control channel.\n", passControl); if (write(fdControl,passControl,strlen(passControl)+1) < 0) { fprintf(stderr,"oxCreateClient(): failed to send passControl.\n"); return(NULL); } } if (passData != NULL) { if (v) fprintf(stderr,"Sending password %s for the data channel.\n", passData); if (write(fdStream,passData,strlen(passData)+1) < 0) { fprintf(stderr,"oxCreateClient(): failed to send passData.\n"); return(NULL); } } controlByteOrder = oxSetByteOrder(fdControl); if (v) fprintf(stderr,"Byte order for control process is %s.\n", (controlByteOrder == 0? "network byte order": (controlByteOrder == 1? "little indican": "big indian"))); engineByteOrder = oxSetByteOrder(fdStream); if (v) fprintf(stderr,"Byte order for engine process is %s.\n", (engineByteOrder == 0? "network byte order": (engineByteOrder == 1? "little indican": "big indian"))); client = (oxclientp) mymalloc(sizeof(oxclient)); oxInitClient(client); client->datafp2 = fp2open(fdStream); if (client->datafp2 == NULL) { fprintf(stderr,"oxCreateClient(): fp2open(fd) failed.\n"); return(NULL); } client->dataport = portStream; client->controlport = portControl; client->controlfd = fdControl; client->id = oxGetClientID(); client->type = CLIENT_SOCKET; /* socket */ client->engineByteOrder = engineByteOrder; client->controlByteOrder = controlByteOrder; return(client); } oxclientp oxCreateClientFile(char *fname,char *mode,char *controlName,char *cmode) { static int clnum = 0x8000; int v = 0; int fdControl = -1; int fdStream = -1; oxclientp client; v = 1; if (strcmp(mode,"w") == 0) { fdStream = creat(fname,S_IWRITE|S_IREAD|S_IRGRP|S_IROTH); if (fdStream < 0) { fprintf(stderr,"\nCreat failed for %s\n",fname); return(NULL); } }else if (strcmp(mode,"r")==0) { fdStream = open(fname,O_RDONLY); if (fdStream < 0) { fprintf(stderr,"\nOpen failed for %s\n",fname); return(NULL); } }else { fprintf(stderr,"\nThe mode %s is not supported.\n",mode); return(NULL); } if (strcmp(cmode,"w") == 0) { fdControl = creat(controlName,S_IWRITE|S_IREAD|S_IRGRP|S_IROTH); if (fdControl < 0) { fprintf(stderr,"\nCreat failed for %s\n",controlName); return(NULL); } }else if (strcmp(cmode,"r")==0) { fdControl = open(controlName,O_RDONLY); if (fdControl < 0) { fprintf(stderr,"\nOpen failed for %s\n",controlName); return(NULL); } }else if (strcmp(cmode,"rw")==0) { fdControl = open(controlName,O_RDWR); if (fdControl < 0) { fprintf(stderr,"\nOpen failed for %s\n",controlName); return(NULL); } }else { fprintf(stderr,"\nThe mode %s is not supported.\n",cmode); return(NULL); } client = (oxclientp) mymalloc(sizeof(oxclient)); oxInitClient(client); client->datafp2 = fp2open(fdStream); if (client->datafp2 == NULL) { fprintf(stderr,"oxCreateClientFile(): fp2open(fd) failed.\n"); return(NULL); } client->dataport = 0; client->controlport = 0; client->controlfd = fdControl; client->id = clnum; clnum++; client->type = CLIENT_FILE; client->engineByteOrder = OX_BYTE_NETWORK_BYTE_ORDER; client->controlByteOrder = OX_BYTE_NETWORK_BYTE_ORDER; return(client); } void oxSendOXheader_generic(int type,int fd,ox_stream ox, int k,int serial) { static int ss = 0; extern int UseOXPacketSerial; if (serial >= 0) ss = serial; else ss++; if (ss < 0) ss=0; if (type == 0) { /* fd */ oxfdSendInt32(fd,k); if (UseOXPacketSerial) oxfdSendInt32(fd,ss); }else { oxSendInt32(ox,k); if (UseOXPacketSerial) oxSendInt32(ox,ss); } } void oxSendOXheader(ox_stream ostream,int k,int serial) { oxSendOXheader_generic(1,-1,ostream,k,serial); } void oxfdSendOXheader(int fd,int k,int serial) { oxSendOXheader_generic(0,fd,(ox_stream) NULL,k,serial); } int oxfdGetOXheader(int fd,int *sss) { char d[4]; int i; int m; for (i=0; i<4; i++) { d[i] = readOneByte(fd); } m = ntohl(* ( (int *)d)); *sss = -1; if (UseOXPacketSerial) { for (i=0; i<4; i++) { d[i] = readOneByte(fd); } *sss = ntohl(* ( (int *)d)); } return(m); } int oxGetOXheader(ox_stream ostream,int *sss) { char d[4]; int m; int i; unlockCtrlCForOx(); /* temporary unlock */ while (fp2select(ostream,1000) == 0) ; restoreLockCtrlCForOx(); for (i=0; i<4; i++) { d[i] = fp2fgetc(ostream); } m = ntohl(* ( (int *)d)); *sss = -1; if (UseOXPacketSerial) { for (i=0; i<4; i++) { d[i] = fp2fgetc(ostream); } *sss = ntohl(* ( (int *)d)); } return(m); } int oxWritePortFile(int func,int port,char *fname) { char name[1024]; FILE *fp; strcpy(name,fname); if (func == 0) { strcat(name,".control"); fp = fopen(name,"w"); fprintf(fp,"%05d\n",port); fclose(fp); }else { strcat(name,".data"); fp = fopen(name,"w"); fprintf(fp,"%05d\n",port); fclose(fp); } } int oxReadPortFile(int func,char *fname) { int port = 0; char name[1024]; FILE *fp; strcpy(name,fname); if (func == 0) { strcat(name,".control"); fp = fopen(name,"r"); {int r; r=fscanf(fp,"%d",&port);} fclose(fp); }else { strcat(name,".data"); fp = fopen(name,"r"); {int r; r=fscanf(fp,"%d",&port);} fclose(fp); } return(port); } char *oxGenPortFile(void) { char *fname; time_t tt; char sstime[512]; fname = (char *)malloc(1024*sizeof(char)); strcpy(fname,getenv("HOME")); strcat(fname,"/.ox."); tt = time(NULL); sprintf(sstime,"%ld",(long) tt); strcat(fname,sstime); if (fname[strlen(fname)-1] == '\n') { fname[strlen(fname)-1] = '\0'; } /* fprintf(stderr,"OxPortFileName=%s\n",fname); */ OxPortFileName = fname; return(fname); } int oxRemovePortFile(void) { char fname[1024]; FILE *fp; SET_MYERROROUT; strcpy(fname,OxPortFileName); strcat(fname,".control"); if ((fp=fopen(fname,"r")) == NULL) { }{ fclose(fp); if (unlink(fname)) { fprintf(MyErrorOut,"fail unlink.\n"); } } strcpy(fname,OxPortFileName); strcat(fname,".data"); if ((fp=fopen(fname,"r")) == NULL) { }{ fclose(fp); if (unlink(fname)) { fprintf(MyErrorOut,"fail unlink.\n"); } } } char *oxGenPass(void) { static int seed = 0; long p; char *s; int i,n; if (seed == 0) { seed = (int) time(NULL) + (int) ((long) &p); srandom((unsigned int) seed); } s = (char *)malloc(128*sizeof(char)); if (s == NULL) { fprintf(stderr,"No more memory.\n"); return(s); } n = (((int)((long) s)) + (int) time(NULL)) % 100; for (i=0; i < n ; i++) random(); p = random(); sprintf(s,"%ld",p); return(s); } static void cancelConnection() { #if defined(__CYGWIN__) extern sigjmp_buf MyEnv_oxmisc; #else extern jmp_buf MyEnv_oxmisc; #endif mysignal(SIGALRM,SIG_IGN); fprintf(stderr,"Time out in TCP/IP connection.\n"); #if defined(__CYGWIN__) MYSIGLONGJMP(MyEnv_oxmisc,1); #else MYLONGJMP(MyEnv_oxmisc,1); #endif } oxclientp oxCreateClient2(int fdstream,int portStream, int fdcontrol,int portControl,int ipmask, char *passControl, char *passData) { int v = 0; int fdControl = -1; int fdStream = -1; int m; char *s; oxclientp client; #if defined(__CYGWIN__) extern sigjmp_buf MyEnv_oxmisc; #else extern jmp_buf MyEnv_oxmisc; #endif int controlByteOrder, engineByteOrder; v = !Quiet; #if defined(__CYGWIN__) if (MYSIGSETJMP(MyEnv_oxmisc,1)) { #else if (MYSETJMP(MyEnv_oxmisc)) { #endif return(NULL); }else{ } alarm((unsigned int) 20); /* setup timeout. */ mysignal(SIGALRM,cancelConnection); switch(ipmask) { case 0:/* only local */ fdControl = socketAcceptLocal(fdcontrol); fdStream = socketAcceptLocal(fdstream); break; default:/* any */ fdControl = socketAccept(fdcontrol); fdStream = socketAccept(fdstream); break; } if (v) fprintf(stderr,"\nControl port %d : Connected.\n",portControl); if (v) fprintf(stderr,"\nStream port %d : Connected.\n",portStream); if (fdStream == -1 || fdControl == -1) { fprintf(stderr,"\nOpen error in oxCreateClient2.\n"); fprintf(stderr,"fdStream=%d, fdControl=%d\n",fdStream,fdControl); return(NULL); } /* Authentication by password. */ m = strlen(passControl)+strlen(passData); if (m > 0) { s = (char *)mymalloc(sizeof(char)*(m+1)); m = strlen(passControl); s[0] = 0; {int r; r=read(fdControl,s,m+1);} s[m] = '\0'; if (strcmp(s,passControl) != 0) { fprintf(stderr,"s=%s, passControl=%s\n",s,passControl); fprintf(stderr,"oxCreateClient2(): password authentication failed for control channel.\n"); close(fdControl); return(NULL); } m = strlen(passData); s[0] = 0; {int r; r=read(fdStream,s,m+1);} s[m] = '\0'; if (strcmp(s,passData) != 0) { fprintf(stderr,"s=%s, passData=%s\n",s,passData); fprintf(stderr,"oxCreateClient2(): password authentication failed for data channel.\n"); close(fdStream); return(NULL); } } mysignal(SIGALRM,SIG_IGN); controlByteOrder = oxSetByteOrder(fdControl); if (v) fprintf(stderr,"Byte order for control process is %s.\n", (controlByteOrder == 0? "network byte order": (controlByteOrder == 1? "little indican": "big indian"))); engineByteOrder = oxSetByteOrder(fdStream); if (v) fprintf(stderr,"Byte order for engine process is %s.\n", (engineByteOrder == 0? "network byte order": (engineByteOrder == 1? "little indican": "big indian"))); client = (oxclientp) mymalloc(sizeof(oxclient)); oxInitClient(client); client->datafp2 = fp2open(fdStream); if (client->datafp2 == NULL) { fprintf(stderr,"oxCreateClient2(): fp2open(fd) failed.\n"); return(NULL); } client->dataport = portStream; client->controlport = portControl; client->controlfd = fdControl; client->id = oxGetClientID(); client->type = CLIENT_SOCKET; /* socket */ client->engineByteOrder = engineByteOrder; client->controlByteOrder = controlByteOrder; return(client); } int oxSetByteOrder(int fd) { char data[1]; int peertype; /* It is for client. read and next write. */ /* oxSocketSelect0(fd,10); wait. */ {int r; r=read(fd,data,1);} peertype = (unsigned char) data[0]; /* We support only Network byte order */ data[0] = OX_BYTE_NETWORK_BYTE_ORDER; {int r; r=write(fd,data,1);} return(OX_BYTE_NETWORK_BYTE_ORDER); } int oxTellMyByteOrder(int fdOut, int fdIn) { char data[1]; int peertype; /* It is for server. read and next write. */ /* We support only Network byte order */ data[0] = OX_BYTE_NETWORK_BYTE_ORDER; {int r; r=write(fdOut,data,1);} fsync(fdOut); /* returns 0 if normal. Does it work for socket? */ {int r; r=read(fdIn,data,1);} /* Read pear's byte order */ return(OX_BYTE_NETWORK_BYTE_ORDER); } struct object OxClientList[MAX_N_OF_CLIENT]; int OxClientListn = 0; int oxGetClientID() { extern struct object OxClientList[]; extern int OxClientListn; extern struct object Nobj; int i; for (i=0; i