⭐ 欢迎来到虫虫下载站! | 📦 资源下载 📁 资源专辑 ℹ️ 关于我们
⭐ 虫虫下载站

📄 ipc.c

📁 小型通讯程序的C语言源程序的C语言源程序库
💻 C
字号:
/* * This file is part of the Minicom Communications Program, * written by Miquel van Smoorenburg 1991/1992/1993. * * ipc.c - talk to the keyserv process. *  * Entry point: * * keyserv(command, arg) *	command can be KINSTALL, KUNINSTALL, or a command for keyserv. *	arg is a 1 byte argument. * */#include <sys/types.h>#if defined (_POSIX_SOURCE) || defined(_BSD43)#  include <unistd.h>#  include <stdlib.h>#endif#undef NULL#include <signal.h>#include <fcntl.h>#include <sys/stat.h>#include <sys/time.h>#include <string.h>#include <stdio.h>#include <setjmp.h>#include <errno.h>#include "window.h"#include "minicom.h"#include "keyboard.h"#include "vt100.h"#include "configsym.h"#ifndef _SELECTstatic int tokeyserv, fromkeyserv;	/* File desciptors of ipc pipes */static int keypid;			/* pid of keyserv  		*/static jmp_buf ackjmp;			/* To jump to after ACK signal  */static int waiting = 0;			/* Do we expect an ACK signal?	*/static int keypress = -1;		/* A key was pressed.		*//* * We got the "ksigio" signal. This means that CTRL-A was pressed in * the main terminal loop, or any key if requested by keyserv(KSIGIO). *//*ARGSUSED*/static void ksigio(dummy)int dummy;{  unsigned char c[8];  int n;  signal(HELLO, ksigio);  while ((n = read(fromkeyserv, c, 8)) < 0 && errno == EINTR)  	errno = 0;  keypress = n ? c[n - 1] : -1;}/* * After we have sent a signal to "keyserv", we wait for it to signal * us back. Otherwise "keyserv" would be swamped with signals and * die ungracefully.... */static void sigack(dummy)int dummy;{  signal(ACK, sigack);  if (waiting) longjmp(ackjmp, 1);  printf("sigack: unexpected ACK signal &^%%$!! (pardon my French)\r\n");}/* * Install the keyserv process. This involves setting up the communications * channels (pipes) and forking the child process. */static void kinstall(){#ifdef _COH3  char mpid[8];#endif  int pipe1[2], pipe2[2];  char buf[2];  char prog[128];  if (pipe(pipe1) < 0 || pipe(pipe2) < 0)  	leave("minicom: out of file descriptors\n");  tokeyserv = pipe1[1];  fromkeyserv = pipe2[0];  /* Set signal handler */  signal(HELLO, ksigio);  signal(ACK, sigack);#ifdef _COH3  sprintf(mpid, "%d", getpid());#endif  switch(keypid = fork()) {  	case -1:  		leave("minicom: could not fork.\n");  		break;  	case 0: /* Child */  		/* Set up fd #1 : stdout */  		dup2(portfd, 1);  		close(portfd);		/* Set up fd #3 : minicom ---> keyserv */		dup2(pipe1[0], 3);		/* Set up fd #4 : minicom <--- keyserv */		dup2(pipe2[1], 4);		/* Close unused file descriptors */		close(pipe1[1]);		close(pipe2[0]);		sprintf(prog, "%s/keyserv", LIBDIR);#ifdef _COH3  		execl(prog, "keyserv", mpid, (char *)NULL);#else  		execl(prog, "keyserv", (char *)NULL);#endif  		exit(0);  	default: /* Parent */		if (setjmp(ackjmp) == 0) {#ifdef DEBUG			printf("keyserv has PID %d\r\n", keypid);#endif  			sleep(2); /* Wait for keyserv to initialize */			waiting = 1;  			buf[0] = KSTOP;  			write(tokeyserv, buf, 2);  			if (kill(keypid, HELLO) < 0) {  				leave("minicom: could not exec keyserv\n");  			}			/* Do nothing 'till ACK signal */			while(1) pause();		}		waiting = 0;		/* close unused pipes */		close(pipe1[0]);		close(pipe2[1]);		break;  }}/* * Install / tell /de-install "keyserv" program. */int keyboard(cmd, arg)int cmd, arg;{  char ch[2];  int pid, stt;  static int lastcmd = -1;  int c;#ifdef _MINIX  /* Avoid race conditions. Unfortunately, this _creates_ a race   * condition on SYSV (linux!) ....   */  if (lastcmd == KSTOP) m_flush(0);#endif  lastcmd = cmd;  if (cmd == KINSTALL) {  	kinstall();  	return(0);  }  if (cmd == KUNINSTALL) {	close(fromkeyserv);	close(tokeyserv);	(void) kill(keypid, SIGKILL);	pid = m_wait(&stt);	return(0);  }  if (cmd == KGETKEY) {	/* Try to read a key from the keyserv process pipe. */	c = keypress;	keypress = -1;	return(c);  }  if (cmd == KSETESC) {	/* Store this because the code expects it. */	escape = arg;  }  if (setjmp(ackjmp) == 0) {	waiting = 1;	ch[0] = cmd;	ch[1] = arg;	write(tokeyserv, ch, 2);	(void) kill(keypid, HELLO);	/* Do nothing 'till ACK signal */	while(1) pause();  }  waiting = 0;  return(0);}/* Dummy sigalarm handler. */static void dummy(){}/* Wait for I/O to happen. We might get interrupted by keyserv. */int check_io(fd1, fd2, tmout, buf, buflen)int fd1;int fd2;int tmout;char *buf;int buflen;{  int n;  int x = 0;  /* OK, set the alarm if needed. */  signal(SIGALRM, dummy);  if (tmout) alarm((tmout + 500) / 1000);  /* We do a read on the first fd, the second one is always stdin. */  if (keypress < 0) {	if (fd1 >= 0) {		/* Read gets interrupted by keypress or alarm. */		n = read(fd1, buf, 127);		buf[n > 0 ? n : 0] = 0;		if (buflen) *buflen = n;		if (n > 0) x |= 1;	} else		/* Wait for keypress or alarm. */		pause();  }  alarm(0);  if (keypress >= 0) x |= 2;  return(x);}#else/* Check if there is IO pending. */int check_io(fd1, fd2, tmout, buf, buflen)int fd1;int fd2;int tmout;char *buf;int *buflen;{  int n = 0, i;  struct timeval tv;  fd_set fds;  tv.tv_sec = tmout / 1000;  tv.tv_usec = (tmout % 1000) * 1000L;  i = fd1;  if (fd2 > fd1) i = fd2;  FD_ZERO(&fds);  if (fd1 >= 0) FD_SET(fd1, &fds);  if (fd2 >= 0) FD_SET(fd2, &fds);  if (select(i+1, &fds, NULL, NULL, &tv))	n = 1 * (FD_ISSET(fd1, &fds) > 0) + 2 * (FD_ISSET(fd2, &fds) > 0);  /* If there is data put it in the buffer. */  if (buf) {	i = 0;	if ((n & 1) == 1) i = read(fd1, buf, 127);	buf[i > 0 ? i : 0] = 0;	if (buflen) *buflen = i;  }  return(n);}int keyboard(cmd, arg)int cmd, arg;{  switch(cmd) {		case KSTART:	case KSTOP:		break;	case KSIGIO:		break;	case KGETKEY:		return(getch());	case KSETESC:		escape = arg;		break;	case KSETBS:		vt_set(-1, -1, NULL, -1, arg, -1, -1);		break;	case KCURST:		vt_set(-1, -1, NULL, -1, -1, -1, NORMAL);		break;	case KCURAPP:		vt_set(-1, -1, NULL, -1, -1, -1, APPL);		break;	default:		/* The rest is only meaningful if a keyserv runs. */		break;  }  return(0);}#endif

⌨️ 快捷键说明

复制代码 Ctrl + C
搜索代码 Ctrl + F
全屏模式 F11
切换主题 Ctrl + Shift + D
显示快捷键 ?
增大字号 Ctrl + =
减小字号 Ctrl + -