Skip to content

Latest commit

 

History

History
277 lines (223 loc) · 10.8 KB

README.md

File metadata and controls

277 lines (223 loc) · 10.8 KB

Corewar

Corewar is a very special programming game created by D. G. Jones and A. K. Dewdney in 1984.
It consists in gathering around a virtual machine one to four players.
This players will load champions who will fight with processes, in order, among other things, to make sure that they are told they are alive.

Processes run sequentially within the same virtual machine, and from the same memory space.
They can, for example, write to each other in order to corrupt others processes, force others to execute instructions that hurt them, try to recreate on the fly the equivalent software of a Côtes du Rhône 1982, etc ...

The game ends when no more processes are alive. At this point, the winner is the last player to be reported as alive.

This is an alternative version of the original game.
The virtual machine does not work the same way
The assembly language is a bit different of the original Redcode

Original Core War wiki here

This code has only been tested on macOS 10.11+

Setup

1 - Requirements

This project use sdl, sdl_image and sdl_ttf libraries.

In order to get the program compile properly you have to install them.
You can use homebrew to simply do this by typing the following

$ brew install sdl sdl_image sdl_ttf

2 - Compiling

First, clone and pull the submodules

$ git clone https://github.com/bil0u/corewar.git
...
$ git submodule update --init

The main Makefile has several rules:

  • all
    Calls lib, asm and vm rules
  • lib
    Compile only the libft submodule
  • asm
    Compile the libft if needed and build the asm binary
  • vm
    Compile the libft if needed and build the corewar binary
  • champs
    Build the asm if not existing and use it to compile all .s files in the champs directory
  • tests
    Build the asm if not existing and use it to compile all .s files in the tests_resources directory
  • clean
    Delete all .o & .cor files
  • fclean
    Calls the clean rule and delete asm & corewar binaries
  • re
    Calls fclean and all rules
  • norm
    Runs the norminette command on all the sources and includes subdirectories (Works only in 42 clusters)

Parts

1 - Assembler

This is the program that will compile your champions and translate them from text to a bytecode, namely a machine code that will be directly interpreted by the virtual machine.

Usage

$ ./asm [file.s ...]

Syntax

  • .name - Name
    Your champion name. (Mandatory)
  • .comment - Comment
    Your champion description. (Mandatory)
  • label: - Labels
    Can be placed before any instruction, useful to mark some positions in your code.
  • rX - Registers
    Design a register to use. Strictly positive, from 1 to 16.
    r1 refers to register 1
  • X / :label - Indirects
    Refers to a 4 bytes integer in ram from the process actual PC value.
    42 refers to the 4 bytes integer located at PC + 42.
    :label refers to the 4 bytes integer located where label points.
  • %X / %:label - Directs
    Integer values.
    %42 refers to the integer 42.
    %:label refers to the number of bytes separating the actual PC from where label points.

Supported instructions

  • live - [DIR]
    Report the player designated by the first parameter as alive. A player lives only as long as at least one process performs live with his player number.
  • ld - [IND/DIR] [REG]
    Load the first parameter in the register passed as the second parameter. If the value of the first parameter equals to zero, then the carry goes to one, otherwise to zero.
  • st - [REG] [IND/DIR]
    Load the contents of the register passed as the first parameter in the second parameter. If the value of the first parameter equals to zero, then the carry goes to one, otherwise to zero.
  • add - [REG] [REG] [REG]
    Adds the second parameter to the first parameter, and stores the result in the third parameter. If the resulting value equals to zero, then the carry goes to one, otherwise to zero.
  • sub - [REG] [REG] [REG]
    Same as add but with substraction.
  • and - [REG/IND/DIR] [REG/IND/DIR] [DIR]
    Performs a logical AND between the first two parameters and stores the result in the third parameter. If the resulting value equals to zero, then the carry goes to one, otherwise to zero.
  • or - [REG/IND/DIR] [REG/IND/DIR] [DIR]
    Same as and but with logical OR.
  • xor - [REG/IND/DIR] [REG/IND/DIR] [DIR]
    Same as xor but with logical XOR.
  • zjmp - [DIR]
    Jump to the address passed in parameter if the carry equals one. If the carry equals zero, nothing happens and the flow continues normally until the next instruction
  • ldi - [REG/IND/DIR] [REG/DIR] [DIR]
    Load the value at the resulting address of the addition of the first two parameters, in the register passed in third parameter. If this value is zero, then the carry goes to one, otherwise to zero.
  • sti - [REG] [REG/IND/DIR] [REG/DIR]
    Load the value contained in the register passed first parameter to the resulting address of the addition of the last two paramametres. If this value is zero, then the carry goes to one, otherwise to zero.
  • fork - [DIR]
    Generate a new process to the address passed in parameter by copy of the calling process. The new process thus keeps of all the registers and the carry, only the PC differs (except in the case of a fork% 0).
  • lld - [IND/DIR] [REG]
    Same as ld but without restriction of addressing.
  • lldi - [REG/IND/DIR] [REG/DIR] [DIR]
    Same as ldi but without restriction of addressing.
  • lfork - [DIR]
    Same as fork but without restriction of addressing.
  • aff - [REG]
    Display on the screen the char corresponding to the value of the register passed in parameter, modulo 256. If this char is NUL, then the carry goes to 1, if not to 0.

Legend
[REG] - Registers
[IND] - Indirects
[DIR] - Directs

See cpu_instructions.h for details

Basic living program example

.name		"Example"
.comment	"I'm a really basic program"

start:		sti r1, %:live, %1
		and r1, %0, r1

live:		live %1
		zjmp %:live

2 - Virtual machine

This is the arena in which the champions will fight. It offers many features, all of them are useful in the battle. It goes without saying that it makes it possible to execute several processes simultaneously.

Usage

$ ./corewar [-a] [-z] [-S N] [-v N] [-D N] [-d N -s N | -V --stealth] [[-n N] file.cor ...]

Options

  • -a
    Prints output of aff instruction (hidden by default)
  • -z
    Reproduces the same bugs as the reference VM given @ 42
  • -n N
    Manually set the player number of the following champion
  • -S N
    Set the execution speed limit at N cycles/seconds
  • -v N
    Set the verbose level. You can add the values. -v 19 will print the lives, cycles and PC movement informations.
    0 - Show only essentials (default)
    1 - Show lives
    2 - Show cycles
    4 - Show operations Args are NOT litteral
    8 - Show deaths
    16 - Show PC movements
  • -D N (See lower important note)
    This mode uses many system calls and slows the program a lot, use with care.
    Set the verbose level. You can add the values. -D 19 will print the game infos, the arena and the processes list.
    1 - Show game informations
    2 - Show the arena memory zone
    4 - Show instructions history
    8 - Show instructions details
    16 - Show processes details
    32 - Show processes registers
  • -d N
    Dumps memory after N cycles then exits
  • -s N
    Runs N cycles, dumps, pauses, repeat
  • -V
    SDL output mode

Controls

In order to be catched, keyboard input for controls needs to be done with focus on the terminal window used to launch the vm binary, as it uses read on STDIN to catch events.

  • <space>
    Pause / Resume the game
  • Return
    Execute the next cycle, useful for step-by-step debug

If launched in debug or visualizer mode, the game is paused by default.


IMPORTANT NOTE

Debug mode only works on macOS 10.11+ if app iTerm (Free) is installed. This is because of the script used to create new windows and launch the logging binary.

You can manually launch a log session in a newly opened terminal window by typing the following

$ <main_directory>/libft/log_server /tmp/libft_log.<window_name> [-scl]
  • <main_directory>
    This is the path where you cloned this project

  • <window_name>

    Name to type Debug level
    inf 1
    mem 2
    ins 4
    arg 8
    job 16
    reg 32
  • -s
    Save all printed output in a log file, in the <main_directory>/libft/log_files directory

  • -c
    Close the terminal window when the log_server stops

  • -l
    Keep the log_server running even if the parent program is terminated
    This allow you to use the same debug windows for your game sessions, and not to have to resize them each time you launch the game

To close a window properly, you must quit the log_server by pressing Ctrl + C first.
If you kill the program by any other way, the log server will not suppress the used fifo, and you will encouter problems for launching a new session. (Your programm will stuck do nothing)

If this happens, you can reset the logging fifos by typing

$ rm -f /tmp/libft_log.*

3 - Tests resources

The references binaries are present as asm_zaz & corewar_zaz.
Theses were the given resources to imitate.

There's also a script called vm_diff.sh that performs a diff with the maximum verbose level beetween the reference binary and the freshly compiled one.

Usage is

$ ./vm_diff.sh [dump cycles] [nb_lines to cmp] [file.cor ...]

Some screenshots

  • Visualizer start screen
    Visualizer

  • Debug level 1 : Game status
    Game status

  • Debug level 2 : Arena content
    Arena

  • Debug level 4 : Instructions history
    Intructions history

  • Debug level 8 : Intructions details
    Intructions details

  • Debug level 16 : Processes list & infos
    Processes

  • Debug level 36 : Processes registers
    Registers

License

This project is licensed under the GNU General Public License v3.0 - see the LICENSE.txt file for details