Skip to content
C++ Makefile Shell
Branch: master
Clone or download
Fetching latest commit…
Cannot retrieve the latest commit at this time.
Permalink
Type Name Latest commit message Commit time
Failed to load latest commit information.
docs
Doxyfile
LICENSE
build-run.sh
builtins.h
colormod.h
main.cpp
makefile
readme.md
shell.h

readme.md

tss - Tom's Simple Shell

This is learning project released in public domain. Comments and improvement suggestions are more than welcome, leave them here.

For the project I used g++, make and doxygen. To get those run sudo apt-get install build-essential doxygen on apt based systems. To build and start the shell run ./build-run.sh, to build shell and docs run make, to clean everything run make clean clean-docs. To build documentation run make docs, you can find output in the docs folder or here.

Shell in a nutshell

A shell or command-line interpreter is a user interface for access to an operating system's services. In general, operating system shells use either a command-line interface (CLI) or graphical user interface (GUI), depending on a computer's role and particular operation. It is named a shell because it is the outermost layer around the operating system kernel.

Although most users think of the shell as an interactive command interpreter, it is really a programming language in which each statement runs a command. Because it must satisfy both the interactive and programming aspects of command execution, it is a strange language, shaped as much by history as by design. — Brian Kernighan & Rob Pike

In it's essence shell is a Read–Eval–Print Loop (REPL), a simple, interactive computer programming environment that takes single user inputs (i.e. single expressions), evaluates them, and returns the result to the user.

Typical shell workflow is as follows:

  • Startup: read and execute configuration files
  • Wait for user input and interpret (read commands from stdin (interactive/file) and execute them)
    • Parsing
    • Expansion
    • Execution
  • Terminate: executes any shutdown commands, free up any memory, and terminate

Startup

config

System calls

The way our programs talk to the kernel is through system calls. These are operations we can tell the kernel to do on our behalf. Examples include: - opening files (open) - reading and writing to them (read, write) - sending and receiving data over the network. (sendto, recvfrom) - starting programs (execve) - process creating a copy of itself (fork)

In fact, more or less everything that involves our programs interacting with the ‘real world’ involves system calls.

Shell is parent process so it utilizes system call fork, this helps us to isolate commands which is important for fault tolerance. We dont want erroneous command to crash whole shell. The fork creates a new process that is a clone of the existing one, right down to the point in the program that's executing. We can then call exec in the newly created child process, leaving our shell's process around for future interaction.

The return value of fork tells us if we're in the child or the parent: it returns 0 for the child, and the id of the new process in the parent.

To wait for the child process to finish we can use wait system call.

Builtins

Every process has its own its own memory and extra metadata that the kernel manages, but that the process can't change directly. Process can use system calls to change user id (setuid), group id (setgid), priority (setpriority), working directory (chdir).

If you were to call change directory command it would invoke chdir internally, change its working dir and return to the parent process - however parents process working dir would stay unchanged, therefore shell has to have builting cd command that calls chdir system call on shell process itself. The handling of cd is then built in to our shell — it's a builtin.

In bash, export lets us change an environment variable's value in the shell itself. As with cd, because a child cannot modify its parent's environment, export must be a builtin.

See manpages on bash builtins man builtins for more.

Pipeline

In Unix-like computer operating systems, a pipeline is a sequence of processes chained together by their standard streams, so that the output of each process (stdout) feeds directly as input (stdin) to the next one.

Pipes are unidirectional; data flows through the pipeline from left to right.

process1 | process2 | process3

The pipe() system call asks the operating system to construct a new anonymous pipe object. This results in two new, opened file descriptors in the process: the read-only end of the pipe, and the write-only end. The pipe ends appear to be normal, anonymous file descriptors, except that they have no ability to seek.

To avoid deadlock and exploit parallelism, the Unix process with one or more new pipes will then, generally, call fork() to create new processes. Each process will then close the end(s) of the pipe that it will not be using before producing or consuming any data. Alternatively, a process might create a new thread and use the pipe to communicate between them.

Named pipes may also be created using mkfifo() or mknod() and then presented as the input or output file to programs as they are invoked. They allow multi-path pipes to be created, and are especially effective when combined with standard error redirection, or with tee.

In most Unix-like systems, all processes of a pipeline are started at the same time, with their streams appropriately connected, and managed by the scheduler together with all other processes running on the machine. An important aspect of this, setting Unix pipes apart from other pipe implementations, is the concept of buffering: for example a sending program may produce 5000 bytes per second, and a receiving program may only be able to accept 100 bytes per second, but no data is lost. Instead, the output of the sending program is held in the buffer. When the receiving program is ready to read data, then next program in the pipeline reads from the buffer. In Linux, the size of the buffer is 65536 bytes (64KiB).

Terminate

Handle SIGINT https://stackoverflow.com/questions/1641182/how-can-i-catch-a-ctrl-c-event-c

Todo:

Globbing

Compare the output of wc -c * between your shell and your regular shell. Expanding the wildcard * is done by the shell before passing arguments to the child. The glob function in the <glob.h> standard header will be useful.

Running a command in the background

In your regular shell, if you end a command with &, it runs in the background. The command will run, but the shell will not block while it does. Ie, it does not wait on the child process.

Refs:

You can’t perform that action at this time.