CHIP-8

Captura de Pong implementado en CHIP-8

CHIP-8 es un lenguaje de programación interpretado, desarrollado por Joseph Weisbecker. Fue inicialmente usado en los microcomputadores de 8 bits COSMAC VIP y Telmac 1800 a mediados de 1970. Los programas de CHIP-8 corren sobre una máquina virtual de CHIP-8. Esto se hizo así para que los video juegos fueran más fáciles de programar en otros computadores.

Aproximadamente 20 años después el CHIP-8 reapareció, pero esta vez, aparecieron diversos intérpretes para algunos modelos de calculadoras gráficas. Como era de esperar, desde finales de 1980 en adelante, esos dispositivos de mano tienen mucho más poder de cálculo que los microcomputadores de mediados de 1970.

Aplicaciones en CHIP-8

Hay un número no pequeño de videojuegos clásicos portados a CHIP-8, como Pong, Space Invaders, Tetris y Pac-Man. También hay disponible un generador aleatorio de laberintos. Esos programas están en dominio público y se pueden encontrar fácilmente en Internet.

El CHIP-8 hoy en día

Hay una implementación de CHIP-8 para casi todas las plataformas imaginables, así como algunas herramientas de desarrollo. A pesar de eso, solo hay un pequeño número de juegos para CHIP-8.

CHIP-8 tiene un descendiente llamado SCHIP (Super Chip), presentado por Erik Bryntse. En 1990, un intérprete de CHIP-8 llamado CHIP-48 fue hecho para la calculadora gráfica HP-48 de esta forma los juegos podían ser programados más fácilmente. Estas extensiones para CHIP-8 fueron conocidas como SCHIP. Este nuevo intérprete cuenta con una mayor resolución y varios opcodes adicionales que permiten programarlo más fácilmente. Si no fuera por el desarrollo del intérprete de CHIP-48, el CHIP-8 no sería conocido en nuestros días.

El desarrollo que más influyó (el cual popularizó el S/CHIP-8 en otras plataformas) fue el emulador de David Winter, que era un desensamblador y además, elaboró una documentación técnica. Estableció una completa lista de opcodes y otras características que hasta esa fecha estaban sin documentar, y fue distribuida a través de varios foros de aficionados a los emuladores. Muchos de los emuladores listados abajo, tuvieron su punto de inicio en dicha documentación.

Descripción de la Máquina Virtual

Memoria

Las direcciones de memoria del CHIP-8 tienen un rango 200h a FFFh, lo que hacen 3.584 bytes. La razón del porqué la memoria comienza en 200h varía de acuerdo a la máquina. Para el Cosmac VIP y el Telmac 1800, los primeros 512 bytes son reservados para el intérprete. En esa máquinas, los 256 bytes más altos (F00h-FFFh en máquinas de 4K) fueron reservados para el refresco de pantalla, y los 96 bytes anteriores (EA0h-EFFh) fueron reservados para los llamados de la pila, uso interno y otras variables.

Registros

El CHIP-8 tiene 16 registros de 8-bit para datos llamados V0, V1, V2, hasta el VF. El registro VF funciona como un flag o Registro de Estado (Status Register). Se usa como carry flag (cuando se usan instrucciones aritméticas) o como detector de colisiones (cuando se dibujan Sprites).

Existe el registro de direcciones llamado I, tiene 16 bits de ancho y es usado con varios opcodes que involucran operaciones con la memoria. De estos 16 bits, el intérprete solo usa los 12 bits menores ya que los 4 bits mayores se usan para la carga de Fuentes.

La pila o stack

La pila solo se usa para almacenar direcciones que serán usadas luego, al regresar de una subrutina. La versión original 1802 permitía almacenar 48 bytes hacia arriba en 12 niveles de profundidad. Las implementaciones modernas en general tienen al menos 16 niveles.

Timers

El CHIP-8 tiene 2 timers o temporizadores. Ambos corren hacia atrás hasta llegar a 0 y lo hacen a 60 hertz.

  • Timer para Delay: este timer se usado para sincronizar los eventos de los juegos. Este valor puede ser escrito y leído.
  • Timer para Sonido: Este timer es usado para efectos de sonidos. Cuando el valor no es 0, se escucha un beep.

Entrada

La entrada está hecha con un teclado de tipo hexadecimal que tiene 16 teclas en un rango de 0 a F. Las teclas '8', '4', '6' y '2' son las típicas usadas para las direcciones. Se usan 3 opcodes para detectar la entrada. Una se activa si la tecla es presionada, el segundo hace lo mismo cuando la no ha sido presionada y el tercero espera que se presione una tecla. Estos 3 opcodes se almacenan en uno de los registros de datos.

Gráficos y Sonido

La Resolución de Pantalla estándar es de 64×32 píxels, y la profundidad del color es Monocromo (solo 2 colores, en general representado por los colores blanco y negro). Los gráficos se dibujan en pantalla solo mediante Sprites los cuales son de 8 pixels de ancho por 1 a 15 pixels de alto. Si un pixel del Sprite está activo, entonces se pinta el color del respectivo pixel en la pantalla; en cambio, si no lo está, no se hace nada. El indicador de acarreo o carry flag (VF) se pone a 1 si cualquier pixel de la pantalla se borra (se pasa de 1 a 0) mientras un pixel se está pintando.

Como se explicó antes, suena un beep cuando el temporizador de sonido no es 0. Para quien esté desarrollando un emulador o intérprete de Chip-8, debe recordar que el sonido debe ser de un solo tono, quedando la frecuencia de dicho tono a decisión del autor del intérprete.

Tabla de instrucciones

CHIP-8 tiene 35 instrucciones, las cuales tienen un tamaño de 2 bytes. Estos opcodes se listan a continuación, en hexadecimal y con los siguientes símbolos:

  • NNN: Dirección
  • KK: constante de 8-bit
  • N: constante de 4-bit
  • X e Y: registro de 4-bit


Opcode Explicación
0NNN Salta a un código de rutina en NNN. Se usaba en los viejos computadores que implementaban Chip-8. Los intérpretes actuales lo ignoran.
00E0 Limpia la pantalla.
00EE Retorna de una subrutina. Se decrementa en 1 el Stack Pointer (SP). El intérprete establece el Program Counter como la dirección donde apunta el SP en la Pila.
1NNN Salta a la dirección NNN. El intérprete establece el Program Counter a NNN.
2NNN Llama a la subrutina NNN. El intérprete incrementa el Stack Pointer, luego pone el actual PC en el tope de la Pila. El PC se establece a NNN.
3XNN Salta a la siguiente instrucción si VX = NN. El intérprete compara el registro VX con el NN, y si son iguales, incrementa el PC en 2.
4XKK Salta a la siguiente instrucción si VX != KK. El intérprete compara el registro VX con el KK, y si no son iguales, incrementa el PC en 2.
5XY0 Salta a la siguiente instrucción si VX = VY. El intérprete compara el registro VX con el VY, y si no son iguales, incrementa el PC en 2.
6XKK Hace VX = KK. El intérprete coloca el valor KK dentro del registro VX.
7XKK Hace VX = VX + KK. Suma el valor de KK al valor de VX y el resultado lo deja en VX.
8XY0 Hace VX = VY. Almacena el valor del registro VY en el registro VX.
8XY1 Hace VX = VX OR VY.

Realiza un bitwise OR (OR Binario) sobre los valores de VX y VY, entonces almacena el resultado en VX. Un OR binario compara cada uno de los bit respectivos desde 2 valores, y si al menos uno es verdadero (1), entonces el mismo bit en el resultado es 1. De otra forma es 0.

8XY2 Hace VX = VX AND VY.
8XY3 Hace VX = VX XOR VY.
8XY4 Suma VY a VX. VF se pone a 1 cuando hay un acarreo (carry), y a 0 cuando no.
8XY5 VY se resta de VX. VF se pone a 0 cuando hay que restarle un dígito al número de la izquierda, más conocido como "pedir prestado" o borrow, y se pone a 1 cuando no es necesario.
8XY6 Establece VF = 1 o 0 según bit menos significativo de VX. Divide VX por 2.
8XY7 Si VY > VX => VF = 1, sino 0. VX = VY - VX.
8XYE Establece VF = 1 o 0 según bit más significativo de VX. Multiplica VX por 2.
9XY0 Salta a la siguiente instrucción si VX != VY.
ANNN Establece I = NNN.
BNNN Salta a la ubicación V[0]+ NNN.
CXKK Establece VX = un Byte Aleatorio AND KK.
DXYN Pinta un sprite en la pantalla. El intérprete lee N bytes desde la memoria, comenzando desde el contenido del registro I. Y se muestra dicho byte en las posiciones VX, VY de la pantalla.

A los sprites que se pintan se le aplica XOR con lo que está en pantalla. Si esto causa que algún pixel se borre, el registro VF se setea a 1, de otra forma se setea a 0. Si el sprite se posiciona afuera de las coordenadas de la pantalla, dicho sprite se le hace aparecer en el lado opuesto de la pantalla.

EX9E Salta a la siguiente instrucción si valor de VX coincide con tecla presionada.
EXA1 Salta a la siguiente instrucción si valor de VX no coincide con tecla presionada (soltar tecla).
FX07 Establece Vx = valor del delay timer.
FX0A Espera por una tecla presionada y la almacena en el registro.
FX15 Establece Delay Timer = VX.
FX18 Establece Sound Timer = VX.
FX1E Índice = Índice + VX.
FX29 Establece I = VX * largo Sprite Chip-8.
FX33 Guarda la representación de VX en formato humano. Poniendo las centenas en la posición de memoria I, las decenas en I + 1 y las unidades en I + 2
FX55 Almacena el contenido de V0 a VX en la memoria empezando por la dirección I
FX65 Almacena el contenido de la dirección de memoria I en los registros del V0 al VX