Skip to content

Instantly share code, notes, and snippets.

@billykwok
Created January 29, 2016 04:44
Show Gist options
  • Save billykwok/0d91a0a2d18a407976e2 to your computer and use it in GitHub Desktop.
Save billykwok/0d91a0a2d18a407976e2 to your computer and use it in GitHub Desktop.
#include "smallsh.h" /* include file for example */
#include <stdlib.h>
#include <signal.h>
/* program buffers and work pointers */
static char inpbuf[MAXBUF], tokbuf[2*MAXBUF], *ptr = inpbuf, *tok = tokbuf;
int intr_p = 0;
char *prompt = "Command>"; /* prompt */
int fg_pid = 0;
userin(p) /* print prompt and read a line */
char *p;
{
int c, count;
/* initialization for later routines */
ptr = inpbuf;
tok = tokbuf;
/* display prompt */
printf("%s ", p);
for(count = 0;;){
if((c = getchar()) == EOF)
return(EOF);
if(count < MAXBUF)
inpbuf[count++] = c;
if(c == '\n' && count < MAXBUF){
inpbuf[count] = '\0';
return(count);
}
/* if line too long restart */
if(c == '\n'){
printf("smallsh: input line too long\n");
count = 0;
printf("%s ", p);
}
}
}
static char special[] = {' ', '\t', '&', ';', '\n', '\0'};
inarg(c) /* are we in an ordinary argument */
char c;
{
char *wrk;
for(wrk = special; *wrk != '\0'; wrk++)
if(c == *wrk)
return(0);
return(1);
}
gettok(outptr) /* get token and place into tokbuf */
char **outptr;
{
int type;
*outptr = tok;
/* strip white space */
for(;*ptr == ' ' || *ptr == '\t'; ptr++)
;
*tok++ = *ptr;
switch(*ptr++){
case '\n':
type = EOL; break;
case '&':
type = AMPERSAND; break;
case ';':
type = SEMICOLON; break;
default:
type = ARG;
while(inarg(*ptr))
*tok++ = *ptr++;
}
*tok++ = '\0';
return(type);
}
void handle_int(int s) {
int c;
if(!fg_pid) {
/* ctrl-c hit at shell prompt */
return;
}
if(intr_p) {
printf("\ngot it, signalling\n");
kill(fg_pid, SIGTERM);
fg_pid = 0;
} else {
printf("\nignoring, type ^C again to interrupt\n");
intr_p = 1;
}
}
/* execute a command with optional wait */
runcommand(cline, where)
char **cline;
int where;
{
int pid, exitstat, ret;
/* ignore signal (linux) */
struct sigaction sa_ign, sa_conf;
sa_ign.sa_handler = SIG_IGN;
sigemptyset(&sa_ign.sa_mask);
sa_ign.sa_flags = SA_RESTART;
sa_conf.sa_handler = handle_int;
sigemptyset(&sa_conf.sa_mask);
sa_conf.sa_flags = SA_RESTART;
if((pid = fork()) < 0){
perror("smallsh");
return(-1);
}
if(pid == 0){
sigaction(SIGINT, &sa_ign, NULL);
sigaction(SIGQUIT, &sa_ign, NULL);
execvp(*cline, cline);
perror(*cline);
exit(127);
} else {
fg_pid = pid;
}
/* code for parent */
/* if background process print pid and exit */
if(where == BACKGROUND){
fg_pid = 0;
printf("[Process id %d]\n", pid);
return(0);
}
/* wait until process pid exits */
sigaction(SIGINT, &sa_conf, NULL);
sigaction(SIGQUIT, &sa_conf, NULL);
while( (ret=wait(&exitstat)) != pid && ret != -1)
;
fg_pid = 0;
return(ret == -1 ? -1 : exitstat);
}
void procline() /* process input line */
{
char *arg[MAXARG+1]; /* pointer array for runcommand */
int toktype; /* type of token in command */
int narg; /* numer of arguments so far */
int type; /* FOREGROUND or BACKGROUND? */
/* reset intr flag */
intr_p = 0;
for(narg = 0;;){ /* loop forever */
/* take action according to token type */
switch(toktype = gettok(&arg[narg])){
case ARG:
if(narg < MAXARG)
narg++;
break;
case EOL:
case SEMICOLON:
case AMPERSAND:
type = (toktype == AMPERSAND) ? BACKGROUND : FOREGROUND;
if(narg != 0){
arg[narg] = NULL;
runcommand(arg, type);
}
if(toktype == EOL)
return;
narg = 0;
break;
}
}
}
main()
{
/* sigaction struct (linux) */
struct sigaction sa;
sa.sa_handler = SIG_IGN;
sigemptyset(&sa.sa_mask);
sa.sa_flags = SA_RESTART;
sigaction(SIGINT, &sa, NULL);
sigaction(SIGQUIT, &sa, NULL);
while(userin(prompt) != EOF)
procline();
}
/* smallsh.h - defs for smallsh command processor */
#include <stdio.h>
#include <unistd.h>
#define EOL 1 /* end of line */
#define ARG 2 /* normal argument */
#define AMPERSAND 3
#define SEMICOLON 4
#define INTERRUPT 5
#define MAXARG 512 /* max. no. command args */
#define MAXBUF 512 /* max. length input line */
#define FOREGROUND 0
#define BACKGROUND 1
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment