CHIP Emulator(3)——OK!

在完成了CHIP8的核心实现后,剩下的事情就是完成画面和输入。为了逻辑区分不干扰,这一部分代码放在main.c函数中实现。其实这部分因为有关GLUT的应用,我也不是太了解,因为我的重点不是放在这上面,所以所做的就是把参考源码依瓢画葫芦,最终实现了CHIP8.

关于GLUT,首先你要进行安装,这里给出Linux下的安装方式GLUT安装

GLUT教程,网上找到了两个,一个是翻译了一部分的,一个原教程。看了以后,main.c中的代码就很容易理解了。对GLUT有兴趣的可以参考下面教程进行深入了解。

教程1:翻译教程
教程2:原教程(英文)

main.c代码:

#include <stdio.h>
#include <GL/glut.h>
#include "mychip8.h"

// Display size
#define SCREEN_WIDTH 64
#define SCREEN_HEIGHT 32

int modifier = 10;

// Window size
// int display_width = SCREEN_WIDTH * modifier;
// int display_height = SCREEN_HEIGHT * modifier;

 int display_width = 640;
 int display_height = 320;

void display();
void reshape_window(GLsizei w, GLsizei h);
void keyboardUp(unsigned char key, int x, int y);
void keyboardDown(unsigned char key, int x, int y);

// Use new drawing method
#define DRAWWITHTEXTURE
//typedef unsigned __int8 u8;

unsigned char screenData[SCREEN_HEIGHT][SCREEN_WIDTH][3]; 
void setupTexture();

int main(int argc, char **argv) 
{       
    if(argc < 2)
    {
        printf("Usage: myChip8.exe chip8application\n\n");
        return 1;
    }

    // Load game
    if(!LoadApp(argv[1]))       
        return 1;

    // Setup OpenGLabclixu123

    glutInit(&argc, argv);          
    glutInitDisplayMode(GLUT_DOUBLE | GLUT_RGBA);

    glutInitWindowSize(display_width, display_height);
    glutInitWindowPosition(320, 320);
    glutCreateWindow("myChip8 by Laurence Muller");

    glutDisplayFunc(display);
    glutIdleFunc(display);
    glutReshapeFunc(reshape_window);        
    glutKeyboardFunc(keyboardDown);
    glutKeyboardUpFunc(keyboardUp); 

#ifdef DRAWWITHTEXTURE
    setupTexture();         
#endif 

    glutMainLoop(); 

    return 0;
}

// Setup Texture
void setupTexture()
{
    int x, y;
    // Clear screen
    for(y = 0; y < SCREEN_HEIGHT; ++y)      
        for(x = 0; x < SCREEN_WIDTH; ++x)
            screenData[y][x][0] = screenData[y][x][1] = screenData[y][x][2] = 0;

    // Create a texture 
    glTexImage2D(GL_TEXTURE_2D, 0, 3, SCREEN_WIDTH, SCREEN_HEIGHT, 0, GL_RGB, GL_UNSIGNED_BYTE, (GLvoid*)screenData);

    // Set up the texture
    glTexParameteri(GL_TEXTURE_2D, GL_TEXTURE_MAG_FILTER, GL_NEAREST);
    glTexParameteri(GL_TEXTURE_2D, GL_TEXTURE_MIN_FILTER, GL_NEAREST);
    glTexParameteri(GL_TEXTURE_2D, GL_TEXTURE_WRAP_S, GL_CLAMP);
    glTexParameteri(GL_TEXTURE_2D, GL_TEXTURE_WRAP_T, GL_CLAMP); 

    // Enable textures
    glEnable(GL_TEXTURE_2D);
}

void updateTexture()
{   
    int x, y;
    // Update pixels
    for( y = 0; y < 32; ++y)        
        for( x = 0; x < 64; ++x)
            if(Gfx[(y * 64) + x] == 0)
                screenData[y][x][0] = screenData[y][x][1] = screenData[y][x][2] = 0;    // Disabled
            else 
                screenData[y][x][0] = screenData[y][x][1] = screenData[y][x][2] = 255;  // Enabled

    // Update Texture
    glTexSubImage2D(GL_TEXTURE_2D, 0 ,0, 0, SCREEN_WIDTH, SCREEN_HEIGHT, GL_RGB, GL_UNSIGNED_BYTE, (GLvoid*)screenData);

    glBegin( GL_QUADS );
        glTexCoord2d(0.0, 0.0);     glVertex2d(0.0,           0.0);
        glTexCoord2d(1.0, 0.0);     glVertex2d(display_width, 0.0);
        glTexCoord2d(1.0, 1.0);     glVertex2d(display_width, display_height);
        glTexCoord2d(0.0, 1.0);     glVertex2d(0.0,           display_height);
    glEnd();
}

// Old gfx code
void drawPixel(int x, int y)
{
    glBegin(GL_QUADS);
        glVertex3f((x * modifier) + 0.0f,     (y * modifier) + 0.0f,     0.0f);
        glVertex3f((x * modifier) + 0.0f,     (y * modifier) + modifier, 0.0f);
        glVertex3f((x * modifier) + modifier, (y * modifier) + modifier, 0.0f);
        glVertex3f((x * modifier) + modifier, (y * modifier) + 0.0f,     0.0f);
    glEnd();
}

void updateQuads()
{
    int x, y;
    // Draw
    for( y = 0; y < 32; ++y)        
        for( x = 0; x < 64; ++x)
        {
            if(Gfx[(y*64) + x] == 0) 
                glColor3f(0.0f,0.0f,0.0f);          
            else 
                glColor3f(1.0f,1.0f,1.0f);

            drawPixel(x, y);
        }
}

void display()
{
    HandleOpcode();

    if(DrawFlag)
    {
        // Clear framebuffer
        glClear(GL_COLOR_BUFFER_BIT);

#ifdef DRAWWITHTEXTURE
        updateTexture();
#else
        updateQuads();      
#endif 

        // Swap buffers!
        glutSwapBuffers();    

        // Processed frame
        DrawFlag = 0;
    }
}

void reshape_window(GLsizei w, GLsizei h)
{
    glClearColor(0.0f, 0.0f, 0.5f, 0.0f);
    glMatrixMode(GL_PROJECTION);
    glLoadIdentity();
    gluOrtho2D(0, w, h, 0);        
    glMatrixMode(GL_MODELVIEW);
    glViewport(0, 0, w, h);

    // Resize quad
    display_width = w;
    display_height = h;
}

void keyboardDown(unsigned char key, int x, int y)
{
    if(key == 27)    // esc
        ;//exit(0);

    if(key == '1')      Keyboard[0x1] = 1;
    else if(key == '2') Keyboard[0x2] = 1;
    else if(key == '3') Keyboard[0x3] = 1;
    else if(key == '4') Keyboard[0xC] = 1;

    else if(key == 'q') Keyboard[0x4] = 1;
    else if(key == 'w') Keyboard[0x5] = 1;
    else if(key == 'e') Keyboard[0x6] = 1;
    else if(key == 'r') Keyboard[0xD] = 1;

    else if(key == 'a') Keyboard[0x7] = 1;
    else if(key == 's') Keyboard[0x8] = 1;
    else if(key == 'd') Keyboard[0x9] = 1;
    else if(key == 'f') Keyboard[0xE] = 1;

    else if(key == 'z') Keyboard[0xA] = 1;
    else if(key == 'x') Keyboard[0x0] = 1;
    else if(key == 'c') Keyboard[0xB] = 1;
    else if(key == 'v') Keyboard[0xF] = 1;

    //printf("Press key %c\n", key);
}

void keyboardUp(unsigned char key, int x, int y)
{
    if(key == '1')      Keyboard[0x1] = 0;
    else if(key == '2') Keyboard[0x2] = 0;
    else if(key == '3') Keyboard[0x3] = 0;
    else if(key == '4') Keyboard[0xC] = 0;

    else if(key == 'q') Keyboard[0x4] = 0;
    else if(key == 'w') Keyboard[0x5] = 0;
    else if(key == 'e') Keyboard[0x6] = 0;
    else if(key == 'r') Keyboard[0xD] = 0;

    else if(key == 'a') Keyboard[0x7] = 0;
    else if(key == 's') Keyboard[0x8] = 0;
    else if(key == 'd') Keyboard[0x9] = 0;
    else if(key == 'f') Keyboard[0xE] = 0;

    else if(key == 'z') Keyboard[0xA] = 0;
    else if(key == 'x') Keyboard[0x0] = 0;
    else if(key == 'c') Keyboard[0xB] = 0;
    else if(key == 'v') Keyboard[0xF] = 0;
}

上述代码需要注意的是原作者给出了两种绘图方式。还有关于CHIP8的键盘,映射方式如下:

Keypad                   Keyboard
+-+-+-+-+                +-+-+-+-+
|1|2|3|C|                |1|2|3|4|
+-+-+-+-+                +-+-+-+-+
|4|5|6|D|                |Q|W|E|R|
+-+-+-+-+       =>       +-+-+-+-+
|7|8|9|E|                |A|S|D|F|
+-+-+-+-+                +-+-+-+-+
|A|0|B|F|                |Z|X|C|V|
+-+-+-+-+                +-+-+-+-+

最后编译生成可执行代码,然后运行游戏应用

./mychip8 pong2.c8

这是一个打乒乓的游戏,运行效果如下,如果没有接着球的话,命令终端就会打印出BEEP!

你可能感兴趣的:(emulator,chip8)