mustard@thor.sdrc.com (Sandy Mustard) (06/01/91)
Hi, I'm having a problem with X on VMS that I'm hoping someone out there might be able to help me with. Basically I have a process that is running as an X client. It spawns using SYS$CREPRC a subprocess (right now I'm using LOGINOUT.EXE) and assigns newly created mailboxes to it's stdin, stdout and stderr. I issue a QIO with a IO$M_WRTATTN mode and have it invoke an AST. This AST will set an event flag whenever some process writes into that mailbox. The event flag is given as the source for the XtAddInput call. The problem is that the call back specified in the XtAddInput call never gets invoked. I do see during the debugging trace that the event flag does get set. Does anyone have any experience with XtAddInput on VMS and could you enlighten me as to how I might fix this? The source code follows if this may help. #include "GlP.h" #include <X11/X.h> #include <X11/Intrinsic.h> #include <iodef.h> #include <dvidef.h> #include <descrip.h> #include <ssdef.h> extern int crtcld(); void post_mbx_ast(); void child_term_ast(); extern FILE * act_file; extern FILE * err_file; static int next_child = 0; struct itmlst /* standard item list structure */ { short len; short code; char *buff; unsigned short *blen; }; /************************************************************************/ /* child output stream callback */ /************************************************************************/ void child_out_cb (gl_child, fdes, id) GL_CHILD *gl_child; int *fdes; int *id; { char *ptr; int stat; ptr = fgets (gl_child->out_bfr, gl_child->out_max-1, gl_child->out_file); if (ptr != NULL) { /* strip off the new line */ gl_child->out_bfr[strlen(gl_child->out_bfr)-1] = '\0'; if (gl_child->out_action != NULL) vcpu (gl_child->out_action); } else { if (gl_child->dead == FALSE) { gl_child->dead = TRUE; if (gl_child->death_action != NULL) vcpu (gl_child->death_action); } XtRemoveInput (*id); stat = SYS$DASSGN(gl_child->out_chan); } } /************************************************************************/ /* child error stream callback */ /************************************************************************/ void child_err_cb (gl_child, fdes, id) GL_CHILD *gl_child; int *fdes; int *id; { char *ptr; int stat; ptr = fgets (gl_child->err_bfr, gl_child->err_max-1, gl_child->err_file); if (ptr != NULL) { /* strip off the new line */ gl_child->err_bfr[strlen(gl_child->err_bfr)-1] = '\0'; if (gl_child->err_action != NULL) vcpu (gl_child->err_action); } else { if (gl_child->dead == FALSE) { gl_child->dead = TRUE; if (gl_child->death_action != NULL) vcpu (gl_child->death_action); } XtRemoveInput (*id); stat = SYS$DASSGN(gl_child->err_chan); } } int child_create () { int i; GL_CHILD * gl_child; int r; gl_child = child_bfr; /* for each child definition */ for (i=0; i<=child_idx; i++) { child_spawn (gl_child); /* add output handler - if requested */ if (gl_child->out_action != NULL) { r = Xt$Add_Input (&gl_child->out_file_en, 0, /* XtInputReadMask, */ child_out_cb, gl_child); } /* add error handler - if requested */ if (gl_child->err_action != NULL) { r = Xt$Add_Input (&gl_child->err_file_en, 0, /* XtInputReadMask, */ child_err_cb, gl_child); } gl_child++; } } /************************************************************************/ /* attach the child to X */ /************************************************************************/ int child_spawn (child) GL_CHILD *child; { int pid; int stat; char mbxnam[32]; struct dsc$descriptor mbxdsc; static char buffer[32]; static int buflen; static int termunit; static struct itmlst dvilst[] = { { sizeof buffer, DVI$_DEVNAM, 0-0, 0-0 }, { sizeof (long), DVI$_UNIT, &termunit, NULL }, { 0, 0, NULL, NULL } }; next_child++; sprintf(buffer, "GL_CLD%08.8d", next_child); /* create a mailbox for messages to the child */ strcpy(mbxnam, buffer); mbxnam[14] ='I'; mbxdsc.dsc$w_length = strlen(mbxnam); mbxdsc.dsc$a_pointer = mbxnam; mbxdsc.dsc$b_class = DSC$K_CLASS_S; mbxdsc.dsc$b_dtype = DSC$K_DTYPE_T; stat = SYS$CREMBX(0, &child->in_chan, BUFSIZ, BUFSIZ, 0, 0, &mbxdsc); dvilst[0].buff = buffer; dvilst[0].blen = &buflen; /* create a pipe for output messages from the child */ if (child->out_action != NULL) { mbxnam[14] ='O'; stat = SYS$CREMBX(0, &child->out_chan, BUFSIZ, BUFSIZ, 0, 0, &mbxdsc); stat = SYS$GETDVIW(0, child->out_chan, 0, dvilst, 0, 0, 0, 0); stat = LIB$GET_EF(&child->out_file_en); stat = SYS$QIO(0, child->out_chan, (IO$_SETMODE|IO$M_WRTATTN), &child->out_iosb, 0, 0, post_mbx_ast, child->out_file_en, 0, 0, 0, 0); child->out_bfr = (char *) calloc(1, child->out_max+1); child->out_file = fopen(buffer, "r"); } /* create a pipe for error messages from the child */ if (child->err_action != NULL) { mbxnam[14] ='E'; stat = SYS$CREMBX(0, &child->err_chan, BUFSIZ, BUFSIZ, 0, 0, &mbxdsc); stat = SYS$GETDVIW(0, child->err_chan, 0, dvilst, 0, 0, 0, 0); stat = LIB$GET_EF(&child->err_file_en); stat = SYS$QIO(0, child->err_chan, (IO$_SETMODE|IO$M_WRTATTN), &child->err_iosb, 0, 0, post_mbx_ast, child->err_file_en, 0, 0, 0, 0); child->err_bfr = (char *) calloc(1, child->err_max+1); child->err_file = fopen(buffer, "r"); } /* create a termination mailbox for the child */ mbxnam[14] ='T'; stat = SYS$CREMBX(0, &child->term_chan, BUFSIZ, BUFSIZ, 0, 0, &mbxdsc); stat = SYS$GETDVIW(0, child->term_chan, 0, dvilst, 0, 0, 0, 0); stat = SYS$QIO(0, child->term_chan, (IO$_SETMODE|IO$M_WRTATTN), &child->iosb, 0, 0, child_term_ast, child, 0, 0, 0, 0); fflush (act_file); /* create the child process */ if (crtcld (child->name, child->path, termunit, child->in_chan, child->out_chan, child->err_chan) == 1) { perror ("child execl"); assert (FALSE); } return(0); } void child_term_ast(child) GL_CHILD * child; { int stat; stat = SYS$QIOW(0, child->out_chan, IO$_WRITEOF, &child->iosb[0], 0, 0, 0, 0, 0, 0, 0, 0); stat = SYS$DASSGN(child->term_chan); } void post_mbx_ast(efn) int efn; { int stat; stat = SYS$SETEF(efn); }
JONESD@kcgl1.eng.ohio-state.edu (David Jones) (06/01/91)
In message <166@thor.sdrc.com>, mustard@thor.sdrc.com (Sandy Mustard) writes: >I'm having a problem with X on VMS that I'm hoping someone out there >might be able to help me with. > >Basically I have a process that is running as an X client. It spawns >using SYS$CREPRC a subprocess (right now I'm using LOGINOUT.EXE) and >assigns newly created mailboxes to it's stdin, stdout and stderr. I >issue a QIO with a IO$M_WRTATTN mode and have it invoke an AST. This >AST will set an event flag whenever some process writes into that mailbox. >The event flag is given as the source for the XtAddInput call. > >The problem is that the call back specified in the XtAddInput call never >gets invoked. I do see during the debugging trace that the event flag >does get set. > > The source code follows if this may help. > > [ source deleted ] You are using LIB$GET_EF to allocate an event flag number, which returns flag numbers in cluster 1 (32-63). The XtAddInput function requires that the flag be in cluster 0 (0-31). You'll have to manage the event flag allocation yourself, I believe the toolkit uses flags 24 and 25 and RMS uses something in the upper 20s, so you should stick with low flag numbers if possible. Also be aware that the client_proc is responsible for clearing the event flag. If it returns with the event flag still set, the dispatcher will call it again immediately. Your code will be more robust if you use the second argument to XtAddInput as well. The GL_CHILD structure should have a psuedo-iosb for use as the second argument to XtAddInput. When you clear the event flag, also zero the first word of the iosb. Likewise, when you set the event flag, deposit a 1 in the first word. The ast parameter for the write attention ast would have to be changed to the address of the structure. David L. Jones | Phone: (614) 292-6929 Ohio State Unviversity | Internet: 1971 Neil Ave. Rm. 406 | jonesd@kcgl1.eng.ohio-state.edu Columbus, OH 43210 | jones-d@eng.ohio-state.edu Disclaimer: A repudiation of a claim.
jon@innerdoor.austin.ibm.com (Jon H. Werner) (06/03/91)
In article <166@thor.sdrc.com>, mustard@thor.sdrc.com (Sandy Mustard) writes: |> |> Hi, |> |> I'm having a problem with X on VMS that I'm hoping someone out there |> might be able to help me with. |> |> Basically I have a process that is running as an X client. It spawns |> using SYS$CREPRC a subprocess (right now I'm using LOGINOUT.EXE) and |> assigns newly created mailboxes to it's stdin, stdout and stderr. I |> issue a QIO with a IO$M_WRTATTN mode and have it invoke an AST. This |> AST will set an event flag whenever some process writes into that mailbox. |> The event flag is given as the source for the XtAddInput call. |> |> The problem is that the call back specified in the XtAddInput call never |> gets invoked. I do see during the debugging trace that the event flag |> does get set. |> You need to get a flag from cluster 0 and use that in the first parameter call to XtAddInput. Set this flag in your write-attention AST procedure. While you're in the AST use a dynamically allocated link-list to store your messages with a ptr to the current message. Then in the XtAddInput procedure read in the message and remove it form the list. I do this because the mail handler can handle input faster than the Event handler in X. This should help you get started. Remember though that VMS 5.4 ASTs are not X re-entrant. -- ______________________________________________________________________________ jon@innerdoor.austin.ibm.com * Insert Standard Disclaimer here * IBM Advanced Workstations Division Austin, Texas USA 512-823-5156