使用内联汇编绘制像素VGA

时间:2016-01-13 03:46:18

标签: c gcc assembly dos vga

我有一个简单的功能,可以使用djgpp和DOS Box中的256 VGA在c中绘制带内联汇编的像素:

byte *VGA = (byte *)0xA0000;   
void plot_pixel(int x, int y, byte color){

  int offset;
  if(x>32 && x <=320 && y>0 && y<180){
    //x=x-1;
    //y=y-1;
    if(x<0){x=0;}
    if(y<0){y=0;}
    offset = (y<<8) + (y<<6) + x;
    VGA[offset]=color;

  }


}

我正在努力将其翻译为内联汇编,我有这个:

void plot_pixel(int x, int y, byte color){

  int offset;
  if(x>32 && x <=320 && y>0 && y<180){
    //x=x-1;
    //y=y-1;
    if(x<0){x=0;}
    if(y<0){y=0;}
    //  offset = (y<<8) + (y<<6) + x;
    //  VGA[offset]=color;

    __asm__ (
            "mov  $0xA000,%edx;"
            "mov  $320,%ax;"
            "mul y;"  //not sure how to reference my variable here
            "add  x,%ax;"  //not sure how to reference my variable here
            "mov %ax,%bx;"
            "mov color,%al;"  //not sure how to reference my variable here
            "mov %al,%bx:(%edx);"


        );
  }


}

但是我在编译器上遇到了几个错误。我不熟悉GCC内联汇编,所以任何纠正我的代码的帮助都会被贬低。

1 个答案:

答案 0 :(得分:2)

首先,对于分段,您必须使用段寄存器(并且您不能将0xA000加载到通用寄存器中并使用它)。然而...

DJGPP有自己的&#34; DOS扩展器&#34;并以32位保护模式运行代码;和段在保护模式下的工作方式非常不同。因此,您无法使用细分,就像您处于实模式一样;并且必须创建一个&#34;段描述符&#34;您可以使用特殊的库函数。有关此示例,请参阅http://www.delorie.com/djgpp/v2faq/faq18_4.html

对于GCC的内联汇编,编译器根本不理解汇编,而且大多只是将代码直接插入到编译器的输出中(可能在进行一些简单的文本替换后)。因此,您需要告诉编译器哪些寄存器用于输入,哪些寄存器用于输出,以及哪些(寄存器,存储器等)是&#34;破坏&#34; (由您的代码修改)。

您应该能够在线找到多个页面,描述如何提供输入/输出/符号列表及其格式。

注意:DJGPP是&#34; GCC移植到DOS&#34;所以GCC的大部分信息对于DJGPP来说都是一样的;并且您将有更多的运气搜索(例如)&#34; GCC内联汇编&#34;而不是你将寻找&#34; DJGPP内联汇编&#34;因为GCC仍在使用中。