home *** CD-ROM | disk | FTP | other *** search
- /* $Header: pd:zvmRCS/simplerexx.c,v 1.2 1993/04/19 18:40:06 rvillari Exp rvillari $ */
- /*
- * Simple ARexx interface by Michael Sinz
- *
- * This is a very "Simple" interface to the world of ARexx...
- * For more complex interfaces into ARexx, it is best that you
- * understand the functions that are provided by ARexx.
- * In many cases they are more powerful than what is presented
- * here.
- *
- * This code is fully re-entrant and self-contained other than
- * the use of SysBase/AbsExecBase and the ARexx RVI support
- * library which is also self-contained...
- */
-
- #include <exec/types.h>
- #include <exec/nodes.h>
- #include <exec/lists.h>
- #include <exec/ports.h>
- #include <exec/memory.h>
-
- #include <proto/exec.h>
-
- #include <rexx/storage.h>
- #include <rexx/rxslib.h>
-
- #include <string.h>
- #include <ctype.h>
-
- int globalARexxReturnCode;
- char globalARexxResult[100];
-
- /*
- * The prototypes for the few ARexx functions we will call...
- */
- struct RexxMsg *CreateRexxMsg(struct MsgPort *,char *,char *);
- void *CreateArgstring(char *,long);
- void DeleteRexxMsg(struct RexxMsg *);
- void DeleteArgstring(char *);
- BOOL IsRexxMsg(struct Message *);
-
- /*
- * Pragmas for the above functions... (To make this all self-contained...)
- * If you use RexxGlue.o, this is not needed...
- *
- * These are for Lattice C 5.x (Note the use of RexxContext->RexxSysBase)
- */
- #pragma libcall RexxContext->RexxSysBase CreateRexxMsg 90 09803
- #pragma libcall RexxContext->RexxSysBase CreateArgstring 7E 0802
- #pragma libcall RexxContext->RexxSysBase DeleteRexxMsg 96 801
- #pragma libcall RexxContext->RexxSysBase DeleteArgstring 84 801
- #pragma libcall RexxContext->RexxSysBase IsRexxMsg A8 801
-
- /*
- * Prototypes for the RVI ARexx calls... (link with RexxVars.o)
- */
- /*
- long __stdargs CheckRexxMsg(struct RexxMsg *);
- long __stdargs GetRexxVar(struct RexxMsg *,char *,char **);
- long __stdargs SetRexxVar(struct RexxMsg *,char *,char *,long);
- */
-
- /*
- * Now, we have made the pragmas needed, let's get to work...
- */
-
- /*
- * A structure for the ARexx handler context
- * This is *VERY* *PRIVATE* and should not be touched...
- */
- struct ARexxContext
- {
- struct MsgPort *ARexxPort; /* The port messages come in at... */
- struct Library *RexxSysBase; /* We will hide the library pointer here... */
- long Outstanding; /* The count of outstanding ARexx messages... */
- char PortName[24]; /* The port name goes here... */
- char ErrorName[28]; /* The name of the <base>.LASTERROR... */
- char Extension[8]; /* Default file name extension... */
- };
-
- #define AREXXCONTEXT struct ARexxContext *
-
- #include "SimpleRexx.h"
-
- /*
- * This function returns the port name of your ARexx port.
- * It will return NULL if there is no ARexx port...
- *
- * This string is *READ ONLY* You *MUST NOT* modify it...
- */
-
- char* ErrorMsg(int r2) {
- switch (r2) {
- case 0 : return "OK";
- case 1 : return "program not found";
- case 2 : return "execution halted";
- case 3 : return "no memory available";
- case 4 : return "character in program";
- case 5 : return "unmatched quote";
- case 6 : return "unterminated comment";
- case 7 : return "clause too long";
- case 8 : return "unrecognized token";
- case 9 : return "symbol or string too long";
-
- case 10 : return "invalid message packet";
- case 11 : return "command string error";
- case 12 : return "error return from function";
- case 13 : return "host environment not found (check server)";
- case 14 : return "required library not found";
- case 15 : return "function not found";
- case 16 : return "no return value";
- case 17 : return "wrong number of arguments";
- case 18 : return "invalid argument to function";
- case 19 : return "invalid PROCEDURE";
-
- case 20 : return "unexpected THEN/ELSE";
- case 21 : return "unexpected WHEN/OTHERWISE";
- case 22 : return "unexpected LEAVE or ITERATE";
- case 23 : return "invalid statement in SELECT";
- case 24 : return "missing THEN clauses";
- case 25 : return "missing OTHERWISE";
- case 26 : return "missing or unexpected END";
- case 27 : return "symbol mismatch on END";
- case 28 : return "invalid DO syntax";
- case 29 : return "incomplete DO/IF/SELECT";
-
- case 30 : return "label not found";
- case 31 : return "symbol expected";
- case 32 : return "string or symbol expected";
- case 33 : return "invalid sub-keyword";
- case 34 : return "required keyword missing";
- case 35 : return "extraneous characters";
- case 36 : return "sub-keyword conflict";
- case 37 : return "invalid template";
- case 38 : return "invalid TRACE request";
- case 39 : return "uninitialized variable";
-
- case 40 : return "invalid variable name";
- case 41 : return "invalid expression";
- case 42 : return "unbalanced parentheses";
- case 43 : return "nesting level exceeded";
- case 44 : return "invalid expression result";
- case 45 : return "expression required";
- case 46 : return "boolean value not 0 or 1";
- case 47 : return "arithmetic conversion error";
- case 48 : return "invalid operand";
- }
- }
-
- int OutstandingMsg(AREXXCONTEXT RexxContext) {
- if (RexxContext) return RexxContext->Outstanding;
- return 0;
- }
-
- char *ARexxName(AREXXCONTEXT RexxContext)
- {
- register char *tmp=NULL;
-
- if (RexxContext) tmp=RexxContext->PortName;
- return(tmp);
- }
-
- /*
- * This function returns the signal mask that the Rexx port is
- * using. It returns NULL if there is no signal...
- *
- * Use this signal bit in your Wait() loop...
- */
- ULONG ARexxSignal(AREXXCONTEXT RexxContext)
- {
- register ULONG tmp=NULL;
-
- if (RexxContext) tmp=1L << (RexxContext->ARexxPort->mp_SigBit);
- return(tmp);
- }
-
- /*
- * This function returns a structure that contains the commands sent from
- * ARexx... You will need to parse it and return the structure back
- * so that the memory can be freed...
- *
- * This returns NULL if there was no message...
- */
- struct RexxMsg *GetARexxMsg(AREXXCONTEXT RexxContext)
- {
- register struct RexxMsg *tmp=NULL;
- register short flag;
-
- if (RexxContext)
- if (tmp=(struct RexxMsg *)GetMsg(RexxContext->ARexxPort))
- {
- if (tmp->rm_Node.mn_Node.ln_Type==NT_REPLYMSG)
- {
- /*
- * If we had sent a command, it would come this way...
- *
- * Since we don't in this simple example, we just throw
- * away anything that looks "strange"
- */
- flag=FALSE;
- if (tmp->rm_Result1) {
- flag=TRUE;
- }
-
- globalARexxReturnCode = tmp->rm_Result1;
-
- if (!tmp->rm_Result1 && tmp->rm_Result2) {
- strncpy(globalARexxResult, (char*)tmp->rm_Result2, sizeof(globalARexxResult));
- DeleteArgstring((char*)tmp->rm_Result2);
- } else if (tmp->rm_Result1 && tmp->rm_Result2) {
- strncpy(globalARexxResult, ErrorMsg(tmp->rm_Result2), sizeof(globalARexxResult));
- } else {
- strcpy(globalARexxResult, "Done");
- }
- globalARexxResult[sizeof(globalARexxResult) - 1] = '\0'; /* ensure null termination */
-
- /*
- * Free the arguments and the message...
- */
- DeleteArgstring(tmp->rm_Args[0]);
- DeleteRexxMsg(tmp);
- RexxContext->Outstanding-=1;
-
- /*
- * Return the error if there was one...
- */
- tmp=flag ? REXX_RETURN_ERROR : NULL;
- tmp = REXX_RETURN_ERROR; // always return an error
- }
- }
- return(tmp);
- }
-
- /*
- */
-
- void* GetReplyARexxMsg(AREXXCONTEXT RexxContext)
- {
- register struct RexxMsg *tmp=NULL;
-
- if (RexxContext)
- if (tmp=(struct RexxMsg *)GetMsg(RexxContext->ARexxPort))
- {
- if (tmp->rm_Node.mn_Node.ln_Type==NT_REPLYMSG)
- {
- /*
- * If we had sent a command, it would come this way...
- */
- /*
- * Free the arguments and the message...
- */
- globalARexxReturnCode = tmp->rm_Result1;
-
- if (!tmp->rm_Result1 && tmp->rm_Result2) {
- strncpy(globalARexxResult, (char*)tmp->rm_Result2, sizeof(globalARexxResult));
- DeleteArgstring((char*)tmp->rm_Result2);
- } else if (tmp->rm_Result1 && tmp->rm_Result2) {
- strncpy(globalARexxResult, ErrorMsg(tmp->rm_Result2), sizeof(globalARexxResult));
- } else {
- strcpy(globalARexxResult, "");
- }
- globalARexxResult[sizeof(globalARexxResult) - 1] = '\0'; /* ensure null termination */
-
- // printf("result1 = %d, result2 = %s\n", globalARexxReturnCode, globalARexxResult);
- DeleteArgstring(tmp->rm_Args[0]);
- DeleteRexxMsg(tmp);
- RexxContext->Outstanding-=1;
- return tmp; // warning: the return value should just be used as a handle that
- // may be used to check whether an outgoing msg. has come back yet
- } else {
- ReplyARexxMsg(RexxContext, tmp, "This port is not for messages", 20);
- return REXX_RETURN_ERROR;
- }
- }
- return 0;
- }
-
- /*
- * Use this to return a ARexx message...
- *
- * If you wish to return something, it must be in the RString.
- * If you wish to return an Error, it must be in the Error.
- * If there is an error, the RString is ignored.
- */
- void ReplyARexxMsg(AREXXCONTEXT RexxContext,struct RexxMsg *rmsg,
- char *RString,LONG Error)
- {
- if (RexxContext) if (rmsg) if (rmsg!=REXX_RETURN_ERROR)
- {
- rmsg->rm_Result2=0;
- if (!(rmsg->rm_Result1=Error))
- {
- /*
- * if you did not have an error we return the string
- */
- if (rmsg->rm_Action & (1L << RXFB_RESULT)) if (RString)
- {
- rmsg->rm_Result2=(LONG)CreateArgstring(RString,
- (LONG)strlen(RString));
- }
- }
-
- /*
- * Reply the message to ARexx...
- */
- ReplyMsg((struct Message *)rmsg);
- }
- }
-
- /*
- * This function will set an error string for the ARexx
- * application in the variable defined as <appname>.LASTERROR
- *
- * Note that this can only happen if there is an ARexx message...
- *
- * This returns TRUE if it worked, FALSE if it did not...
- */
- short SetARexxLastError(AREXXCONTEXT RexxContext,struct RexxMsg *rmsg,
- char *ErrorString)
- {
- register short OkFlag=FALSE;
-
- if (RexxContext) if (rmsg) if (CheckRexxMsg((struct Message*)rmsg))
- {
- /*
- * Note that SetRexxVar() has more than just a TRUE/FALSE
- * return code, but for this "basic" case, we just care if
- * it works or not.
- */
- if (!SetRexxVar((struct Message*)rmsg,RexxContext->ErrorName,ErrorString,
- (long)strlen(ErrorString)))
- {
- OkFlag=TRUE;
- }
- }
- return(OkFlag);
- }
-
- /*
- * This function will send a string to ARexx...
- *
- * The default host port will be that of your task...
- *
- * If you set StringFile to TRUE, it will set that bit for the message...
- *
- * Returns TRUE if it send the message, FALSE if it did not...
- */
- void* SendARexxMsg(AREXXCONTEXT RexxContext,char *RString,
- short StringFile)
- {
- register struct MsgPort *RexxPort;
- register struct RexxMsg *rmsg;
- register void* flag=0;
-
- if (RexxContext) if (RString)
- {
- if (rmsg=CreateRexxMsg(RexxContext->ARexxPort,
- RexxContext->Extension,
- RexxContext->PortName))
- {
- rmsg->rm_Action=RXCOMM | (StringFile ?
- (1L << RXFB_STRING):0) |
- (1L << RXFB_RESULT);
- if (rmsg->rm_Args[0]=CreateArgstring(RString,
- (LONG)strlen(RString)))
- {
- /*
- * We need to find the RexxPort and this needs
- * to be done in a Forbid()
- */
- Forbid();
- if (RexxPort=FindPort(RXSDIR))
- {
- /*
- * We found the port, so put the
- * message to ARexx...
- */
- PutMsg(RexxPort,(struct Message *)rmsg);
- RexxContext->Outstanding+=1;
- flag=rmsg;
- }
- else
- {
- /*
- * No port, so clean up...
- */
- DeleteArgstring(rmsg->rm_Args[0]);
- DeleteRexxMsg(rmsg);
- }
- Permit();
- }
- else DeleteRexxMsg(rmsg);
- }
- }
- return(flag);
- }
-
- /*
- * This function closes down the ARexx context that was opened
- * with InitARexx...
- */
- void FreeARexx(AREXXCONTEXT RexxContext)
- {
- struct RexxMsg *rmsg;
-
- if (RexxContext)
- {
- /*
- * Clear port name so it can't be found...
- */
- RexxContext->PortName[0]='\0';
-
- /*
- * Clean out any outstanding messages we had sent out...
- */
- while (RexxContext->Outstanding)
- {
- WaitPort(RexxContext->ARexxPort);
- while (rmsg=GetARexxMsg(RexxContext))
- {
- if (rmsg!=REXX_RETURN_ERROR)
- {
- /*
- * Any messages that come now are blown
- * away...
- */
- SetARexxLastError(RexxContext,rmsg,
- "99: Port Closed!");
- ReplyARexxMsg(RexxContext,rmsg,
- NULL,100);
- }
- }
- }
-
- /*
- * Clean up the port and delete it...
- */
- if (RexxContext->ARexxPort)
- {
- while (rmsg=GetARexxMsg(RexxContext))
- {
- /*
- * Any messages that still are coming in are
- * "dead" We just set the LASTERROR and
- * reply an error of 100...
- */
- SetARexxLastError(RexxContext,rmsg,
- "99: Port Closed!");
- ReplyARexxMsg(RexxContext,rmsg,NULL,100);
- }
- DeletePort(RexxContext->ARexxPort);
- }
-
- /*
- * Make sure we close the library...
- */
- if (RexxContext->RexxSysBase)
- {
- CloseLibrary(RexxContext->RexxSysBase);
- }
-
- /*
- * Free the memory of the RexxContext
- */
- FreeMem(RexxContext,sizeof(struct ARexxContext));
-
- }
- }
-
- /*
- * This routine initializes an ARexx port for your process
- * This should only be done once per process. You must call it
- * with a valid application name and you must use the handle it
- * returns in all other calls...
- *
- * NOTE: The AppName should not have spaces in it...
- * Example AppNames: "MyWord" or "FastCalc" etc...
- * The name *MUST* be less that 16 characters...
- * If it is not, it will be trimmed...
- * The name will also be UPPER-CASED...
- *
- * NOTE: The Default file name extension, if NULL will be
- * "rexx" (the "." is automatic)
- */
- AREXXCONTEXT InitARexx(char *AppName,char *Extension)
- {
- register AREXXCONTEXT RexxContext=NULL;
- register short loop;
- register short count;
- register char *tmp;
- int hasARexx = 0;
-
- if (RexxContext=AllocMem(sizeof(struct ARexxContext),
- MEMF_PUBLIC|MEMF_CLEAR))
- {
- if (RexxContext->RexxSysBase=OpenLibrary("rexxsyslib.library",
- NULL))
- {
- /*
- * Set up the extension...
- */
- if (!Extension) Extension="rexx";
- tmp=RexxContext->Extension;
- for (loop=0;(loop<7)&&(Extension[loop]);loop++)
- {
- *tmp++=Extension[loop];
- }
- *tmp='\0';
-
- /*
- * Set up a port name...
- */
- tmp=RexxContext->PortName;
- for (loop=0;(loop<16)&&(AppName[loop]);loop++)
- {
- *tmp++=toupper(AppName[loop]);
- }
- *tmp='\0';
-
- /*
- * Set up the last error RVI name...
- *
- * This is <appname>.LASTERROR
- */
- strcpy(RexxContext->ErrorName,RexxContext->PortName);
- strcat(RexxContext->ErrorName,".LASTERROR");
-
- /* We need to make a unique port name... */
- Forbid();
- for (count=1,RexxContext->ARexxPort=(VOID *)1;
- RexxContext->ARexxPort;count++)
- {
- stci_d(tmp,count);
- RexxContext->ARexxPort=
- FindPort(RexxContext->PortName);
- }
-
- RexxContext->ARexxPort=CreatePort(
- RexxContext->PortName,NULL);
- if (FindPort(RXSDIR)) hasARexx = 1;
- else hasARexx = 0;
- Permit();
- }
-
- if ( (!(RexxContext->RexxSysBase))
- || (!(RexxContext->ARexxPort))
- || (!hasARexx)
- )
- {
- FreeARexx(RexxContext);
- RexxContext=NULL;
- }
- }
- return(RexxContext);
- }
-
- /*
- * This routine initializes an ARexx port for your process
- * This should only be done once per process. You must call it
- * with a valid application name and you must use the handle it
- * returns in all other calls...
- *
- * NOTE: The AppName should not have spaces in it...
- * Example AppNames: "MyWord" or "FastCalc" etc...
- * The name *MUST* be less that 16 characters...
- * If it is not, it will be trimmed...
- * The name will also be UPPER-CASED...
- *
- * NOTE: The Default file name extension, if NULL will be
- * "rexx" (the "." is automatic)
- */
-
- AREXXCONTEXT InitARexxSingleApp(char *AppName,char *Extension)
- {
- register AREXXCONTEXT RexxContext=NULL;
- register short loop;
- register short count;
- register char *tmp;
- int hasARexx = 0;
-
- if (RexxContext=AllocMem(sizeof(struct ARexxContext),
- MEMF_PUBLIC|MEMF_CLEAR))
- {
- if (RexxContext->RexxSysBase=OpenLibrary("rexxsyslib.library",
- NULL))
- {
- /*
- * Set up the extension...
- */
- if (!Extension) Extension="rexx";
- tmp=RexxContext->Extension;
- for (loop=0;(loop<7)&&(Extension[loop]);loop++)
- {
- *tmp++=Extension[loop];
- }
- *tmp='\0';
-
- /*
- * Set up a port name...
- */
- tmp=RexxContext->PortName;
- for (loop=0;(loop<16)&&(AppName[loop]);loop++)
- {
- *tmp++=toupper(AppName[loop]);
- }
- *tmp='\0';
-
- /*
- * Set up the last error RVI name...
- *
- * This is <appname>.LASTERROR
- */
- strcpy(RexxContext->ErrorName,RexxContext->PortName);
- strcat(RexxContext->ErrorName,".LASTERROR");
-
- /* We need to make a unique port name... */
- Forbid();
-
- RexxContext->ARexxPort=
- FindPort(RexxContext->PortName);
-
- if (!RexxContext->ARexxPort) {
- RexxContext->ARexxPort=CreatePort(
- RexxContext->PortName,NULL);
- if (FindPort(RXSDIR)) hasARexx = 1;
- else hasARexx = 0;
- } else {
- RexxContext->ARexxPort = 0;
- }
-
- Permit();
- }
-
- if ( (!(RexxContext->RexxSysBase))
- || (!(RexxContext->ARexxPort))
- || (!hasARexx)
- )
- {
- FreeARexx(RexxContext);
- RexxContext=NULL;
- }
- }
- return(RexxContext);
- }
-
-