dbin@norsat.UUCP (Dave Binette) (09/22/89)
Several weeks ago I posted a request to USENET asking for code fragments of programs that used interprocess communications. The response was encouraging, thank you all for the help. In return I am submitting this for your amusement/edification. A small low overhead multiuser chat program for 16 users using shared memory. It uses an algorithm I've never seen anywhere else, but then... I haven't been looking for chat algorithms. Its *definately* not the best, but it works. Theoretically it can break and lose a message but I've never seen it happen (yet). ------------------------ chop chop ----------------------------- # This is a shell archive. Remove anything before this line, # then unpack it by saving it in a file and typing "sh file". # # Wrapped by dbin on Fri Sep 22 00:14:33 PDT 1989 # Contents: README makefile memchat.1 memchat.c echo x - README sed 's/^@//' > "README" <<'@//E*O*F README//' memchat 16 user shared memory chat program Released to PUBLIC DOMAIN by David J. Binette in appreciation of the replies to my request for IPC code fragments on the USEers NETwork. I wrote this code to utilize shared memory A single executable called "memchat" is compiled. There is no server. The only tunable parameters in the code are: #define MAXMSG 20 /* # of messages */ #define MSGSIZ 80 /* size of a typed line */ Again, I'd like to thank all those from USENET who responded to my requests for code fragments and help regarding message Q's - uucp: {uunet,ubc-cs}!van-bc!norsat!dbin | 302-12886 78th Ave bbs: (604)597-4361 24/12/PEP/3 | Surrey BC CANADA voice: (604)597-6298 (Dave Binette) | V3W 8E7 @//E*O*F README// chmod u=rw,g=r,o=r README echo x - makefile sed 's/^@//' > "makefile" <<'@//E*O*F makefile//' # compilation for "memchat" under SCO XENIX 2.3.1 # sysname=XENIX # nodename=norsat # release=2.3.1 # version=SysV # machine=i80386 # origin=3 # # The source code for memchat.c, the executable memchat # as well as the accompanying documentation README and memchat.1 # are placed in the public domain. # Use it, sell it, do what you will. # If your nice, you'll leave my name on it. # If your kind, and you modify it, you'll put your name on it too! # Author: uunet!norsat!dbin (David J. Binette) memchat: memchat.c cc -M3e -W3 -Od memchat.c -s -o memchat -lx @//E*O*F makefile// chmod u=rw,g=r,o=r makefile echo x - memchat.1 sed 's/^@//' > "memchat.1" <<'@//E*O*F memchat.1//' @.TH MEMCHAT 1 NORSAT @.SH NAME memchat \- 16 user Chat program using Shared Memory. @.SH SYNOPSIS @.B memchat @.SH DESCRIPTION This program sends and receives data via shared memory. V1.0 Released to PUBLIC DOMAIN by David J. Binette in appreciation of the replies to a request for code fragments on USENET. @.SH EXAMPLES memchat @.RE @.SH SEE ALSO write(C), hello(C), mail(C) @.SH RETURN VALUE Returns 0 for normal successfull exit. @.br Returns 255 if memchat was full. @.br Failed calls return with the errno of the failed function. @.SH NOTES No warranties expressed or implied, use at your own risk. @.SH BUGS The algorithm may break (theoretically) and lose a message. @//E*O*F memchat.1// chmod u=rw,g=r,o=r memchat.1 echo x - memchat.c sed 's/^@//' > "memchat.c" <<'@//E*O*F memchat.c//' /* memchat.c * * 16 user shared memory chat program * Released to public domain by David J. Binette * in appreciation of the replies to my request for * IPC code fragments on the USEers NETwork. * * The algorithm used may break (theoretically) but I've never seen * it happen, its just a matter of timing. At worst, a message is lost. * * The source code for memchat.c, the executable memchat * as well as the accompanying documentation README and memchat.1 * are placed in the public domain. * Use it, sell it, do what you will. * If your nice, you'll leave my name on it. * If your kind, and you modify it, you'll put your name on it too! */ /* * Program Name : memchat * Associated files: memchat.1 makefile memchat.c * Author : uunet!norsat!dbin (David J. Binette) * Version : 1.0 * Release Date : Thu Sep 21 21:38:47 PDT 1989 * Revision History: * : * : * : * : */ #include <stdio.h> #include <sys/types.h> #include <sys/ipc.h> #include <sys/shm.h> #include <sd.h> #include <signal.h> #include <termio.h> #include <sys/ioctl.h> /* ------------------------------------------------------- */ /* you can change this if you want */ #define MAXMSG 20 /* # of messages */ #define MSGSIZ 80 /* size of a typed line */ /* ------------------------------------------------------- */ #define KEY (key_t) 60 #define FNULL (struct COMMON *) 0 #define FAILURE (struct COMMON *) -1 #define FALSE 0 #define TRUE 1 #define EOS '\0' /* ------------------------------------------------------- */ extern int errno; int clean(); int Rval=0; int shmid; unsigned int mynum; char myname[L_cuserid+1]; struct COMMON * base; char Inbuf[MSGSIZ+1]; /* this needn't be global */ /* ------------------------------------------------------- */ extern struct COMMON { unsigned int usr; unsigned int map[MAXMSG]; char txt[MAXMSG][L_cuserid + 2 + MSGSIZ+1]; }; /* ------------------------------------------------------- */ void putmsg(s) char *s; { int i=MAXMSG; while(i==MAXMSG) for(i=0; i<MAXMSG; i++) if(base->map[i] == 0) { base->map[i] = mynum; strcpy(base->txt[i],myname); strcat(base->txt[i],"> "); strcat(base->txt[i],s); base->map[i] = base->usr &~mynum; break; } } /* ------------------------------------------------------- */ int getmsg() { int i; int r=0; for(i=0; i<MAXMSG; i++) if(base->map[i] & mynum) { r=1; puts(base->txt[i]); base->map[i] &= ~mynum; } return(r); } /* ------------------------------------------------------- */ int getaline() { int r; *Inbuf=EOS; if(r=(rdchk(fileno(stdin))>0)) if(fgets(Inbuf,MSGSIZ,stdin)) if(*Inbuf) putmsg(Inbuf); return(r); } /* ------------------------------------------------------- */ void main() { int i; int created=0; Rval=0; if(!cuserid(myname)) strncpy(myname,"unknown",L_cuserid); signal(SIGINT, clean); if((shmid = shmget(KEY, sizeof(struct COMMON), 0)) < 0) { if((shmid = shmget(KEY, sizeof(struct COMMON), IPC_CREAT | 0666)) < 0) { Rval=errno; perror("shmget failed"); exit(Rval); } created=1; } if((base = (struct COMMON *)shmat(shmid, FNULL, 0)) == FAILURE) { Rval=errno; perror("shmat failed"); exit(Rval); } if(created) /* initialize the common area */ { base->usr=0; for(i=0; i<MAXMSG; i++) { base->map[i]=0; base->txt[i][0]=EOS; } } /* get a user # */ for(mynum=0x8000; mynum; mynum>>=1) /* portable? not likely */ { if(!(base->usr & mynum)) break; } if(mynum) { putmsg("Arrives!"); base->usr|=mynum; } else { puts("Sorry, no more users allowed in chat."); Rval=255; clean(); } puts("\nWelcome to memchat V1.0"); puts("Type in your text and press [ENTER]"); puts("Press [DEL] to end session\n"); for( ; ; ) if(!(getaline() | getmsg())) nap(125L); clean(); } /* ------------------------------------------------------- */ clean() { int i; putmsg("Departs!"); for(i=0; i<MAXMSG; i++) base->map[i] &= ~mynum; base->usr &= ~mynum; shmdt(base); shmctl(shmid, IPC_RMID, NULL); exit(Rval); } /* ------------------------------------------------------- */ @//E*O*F memchat.c// chmod u=rw,g=r,o=r memchat.c echo Inspecting for damage in transit... temp=/tmp/shar$$; dtemp=/tmp/.shar$$ trap "rm -f $temp $dtemp; exit" 0 1 2 3 15 cat > $temp <<\!!! 24 119 742 README 19 98 562 makefile 33 112 672 memchat.1 215 529 4244 memchat.c 291 858 6220 total !!! wc README makefile memchat.1 memchat.c | sed 's=[^ ]*/==' | diff -b $temp - >$dtemp if [ -s $dtemp ] then echo "Ouch [diff of wc output]:" ; cat $dtemp else echo "No problems found." fi exit 0 --- "If I was smarter than I was bad, I wouldn't get in any trouble" (Laura my 4 yr. old daughter) uucp: {uunet,ubc-cs}!van-bc!norsat!dbin | 302-12886 78th Ave bbs: (604)597-4361 24/12/PEP/3 | Surrey BC CANADA voice: (604)597-6298 (Dave Binette) | V3W 8E7