zoukankan      html  css  js  c++  java
  • 基于FPGA的DW8051移植(一)

      最近 半个月都在移植8051,看到DW8051内核资料比较齐全又是新思发布的,所以就开始玩弄

      可是这半个月的努力几近白费 —— 移植失败了,不知道从何着手这个内核。可能大家能找到不同的版本,我的是最初的版本,里面包含了datasheet 以及一些说明文档,看上去貌似是最原始的(可能已经修改过了),本想传上来,但是我一直到偶没有找到哪里可以上传文件。

      我没有添加片外memory,只有片内64k的rom 和256k的ram

      首先来看看我的tb

      1 `timescale 1us/100ns 
      2 
      3 module dw8051_tb ; 
      4 
      5 reg         clk ,            reset ;
      6 reg         int0_n,        int1_n ; 
      7 reg      rxd0_in ,   rxd1_in ; 
      8 reg      t0 ,        t1     ;
      9 
     10 wire         rst_out_n ;
     11 wire         stop_mode_n,   idle_mode_n ; 
     12 wire     rxd0_out ,     txd0    ;
     13 wire     rxd1_out ,     txd1    ;
     14 wire     t0_out  ,      t1_out  ;
     15 
     16 integer i ; 
     17 
     18 DW8051_top  cpu_core(
     19 
     20              .clk (clk)         , 
     21              .reset (reset)       ,
     22              .rst_out_n (rst_out_n)    ,           
     23              .stop_mode_n (stop_mode_n) , 
     24              .idle_mode_n (idle_mode_n) ,         
     25                          
     26            //int
     27              .int0_n (int0_n)      , 
     28              .int1_n (int1_n)      , 
     29            
     30            //serial port
     31              .rxd0_in (rxd0_in)     , 
     32              .rxd0_out (rxd0_out)    , 
     33              .txd0 (txd0)        , 
     34                          
     35              .rxd1_in (rxd1_in)      , 
     36              .rxd1_out (rxd1_out)    , 
     37              .txd1 (txd1)        , 
     38            
     39            //timer/counter input/output
     40              .t0 (t0)          , 
     41              .t1 (t1)          , 
     42              .t0_out (t0_out)       , 
     43              .t1_out (t1_out)      , 
     44                          
     45 
     46        //sfr interface
     47                .sfr_data_in () , 
     48              .sfr_addr ()    , 
     49              .sfr_data_out () , 
     50              .sfr_wr ()      , 
     51              .sfr_rd ()      ,
     52            
     53            //external ram interface or user-defined peripheral reg
     54                .mem_data_in () , 
     55              .mem_addr ()    , 
     56              .mem_data_out (), 
     57              .mem_wr_n ()     , 
     58              .mem_rd_n ()
     59             );    
     60 
     61 always #10 clk = ~clk ; 
     62                             
     63 initial   begin 
     64     i = 0  ;
     65     clk = 0 ;     reset =0 ; 
     66     int0_n=0 ;    int1_n=0 ;
     67     rxd0_in=0 ;   rxd1_in=0 ;
     68     t0 =  0  ;    t1  = 0   ;
     69     #180 reset =1 ; 
     70     
     71     
     72     
     73 //**********测试1 : 发现无论如何修改idata初始化空间大小,程序都会进入29-2a-2b-2c 这个循环,无法跳出
     74     // while (i<8'hff) begin 
     75             // @(cpu_core.irom_addr == 16'h0029)   i = i+1 ; 
     76             // if (i>8'h7f) begin 
     77                     // $display ("rom loop at 0029 is over 7f time ");
     78                     // $display ("  i = %d 
    ", i );
     79                     // end 
     80             // end 
     81 //---------------------------------------------------------------------------------------            
     82 
     83 /**********测试2: 依据测试1追查27这个地址上的循环次数数值去了哪里
     84     1,@(cpu_core.core.irom_addr == 16'h0027) 测试发现跑到最后一个displays里面去了 ,屏蔽掉所有的if
     85 打印具体数据信息结果显示 cpu_core.core.i_cpu.biu_instr == 8'h78
     86 而cpu_core.core.i_cpu.biu_data_in == 8'h00
     87     2,@(cpu_core.core.irom_addr == 16'h0028) 测试发现cpu_core.core.i_cpu.biu_instr == 8'h3f而
     88 cpu_core.core.i_cpu.biu_data_in == 8'h00。由1和2对比可知数据有一个周期的停滞期
     89 */
     90 //@(cpu_core.core.irom_addr == 16'h0028) begin 
     91 //        //if (cpu_core.core.int_rom_data_out == 8'h3f) begin 
     92 //                //if (cpu_core.core.i_cpu.biu_instr == 8'h3f)
     93 //                        $display ("the rom data 3f have get to biu_instr = %h  
     ",cpu_core.core.i_cpu.biu_instr );
     94 //                //if (cpu_core.core.i_cpu.biu_data_in == 8'h3f)
     95 //                        $display ("the rom data 3f have get to biu_data_in  = %h
    ", cpu_core.core.i_cpu.biu_data_in);
     96 //                end 
     97 //        else  begin 
     98 //                $display ("operation error ! rom data 3f at address 27 have not load in ");
     99 //                end 
    100 //                end 
    101 
    102 /********测试3 依据测试2追查内核执行情况 
    103 
    104 
    105 */
    106 $display ("display  data  :  
    ");
    107 $monitor ("%h",cpu_core.core.i_cpu.i_opdec.op);
    108 #8000 $stop ; 
    109 
    110 
    111 
    112 
    113     @(cpu_core.irom_addr == 16'h0019)
    114     #80000  $stop ; 
    115 
    116 
    117 
    118 
    119 end 
    120 
    121 endmodule

      最后的一个stop 是始终执行不到那里去,也就是说代码根本没有执行到程序存储地址19这里来。

      当然需要解释一下我为什么要这么测试,不懂为什么tb这么写,可以先忽略掉。我们一步步来追究,首先看看我的c程序

     1 #include <reg51.h>
     2 #define uchar unsigned char 
     3 #define uint unsigned int 
     4 
     5 sbit LED = P0^0 ; 
     6 
     7 void DelayMS (uint x )
     8 {
     9     uchar i ; 
    10     while (x--);
    11     {
    12         for (i=0;i<2;i++) ;
    13         
    14     }
    15 }
    16 
    17 void main ()
    18 {
    19    LED = 1 ; 
    20    while (1) 
    21     {
    22      
    23         DelayMS(5);
    24         LED=~LED ;
    25     }
    26 }

    调试窗口中显示我的程序起始地址是19,也就是说如果执行我的main函数要从19这个地址作为入口进行执行。在后面的step按钮下一步步调试发现,我的子函数起始地址是 03 。整个用户程序的起始地址是03,终止地址是25.

    下面来看看keilC输出的hex

    :03000000020026D5
    :0C002600783FE4F6D8FD75810702001950
    :10000300EF1FAA0670011E4A600BE4FDEDC39402C4
    :0500130050EE0D80F726
    :0100180022C5
    :0D001900D2807F057E00120003B28080F5CA
    :00000001FF

    依据我查到的资料上说: 03000000020026D5

    03 代码数据数目  0000 代表存储地址,00代表数据类型   020026代表数据  D5代表校验码

    依据上述的意思去查QuartusII里面的信息

    经过对比发现两者是一致的,说明QuartusII 对hex档的读取是没有问题的(当然我的C程序也是调试过没有问题的)

    下面让我们看看modelsim仿真出来的结果

    modelsim中仿真可以看到,程序执行地址 00- 01 - 02- 26 -27 - 28 - 29 - 2a - 2b - 2c - 29........然后就是这个循环一直都没有退出,也就是说根本没有执行用户程序。

      继续追踪这个程序段

    reset之后一切明白了,程序从00地址开始执行 转跳到地址26——记住用户程序是从03到25,所以这个时候执行的是startup程序段,现在追踪startup,这个startup.a51是每个单片机上电的时候都需要进行执行的代码段,也就是初始化单片机的程序(具体的操作大家自己找资料看,网上很多人对这个文件有了中文的解析,比如这位大虾的http://www.cnblogs.com/heiyue/p/3262389.html

    在调试中发现

     

     立即数3f给了r0 ,也就是地址段27这个地方的数据,它表示的是我的ram的大小(startup.a51里面默认的是80编译出来是7f,改成40编译出来是3f),也就是我的初始化执行这个loop的次数。

    好,现在可以解释一下为什么有了我的程序中的测试1

    依据程序中规定的,这个loop应该是执行3f次—— 我也可以把这个loop直接停止掉,不用去初始化我的ram。因为我不管怎么修改这个loop都跳转不出来,所以我就开始追踪 3f去哪里了,所以才有了测试2

    测试2发现这个3f居然进了decode模块的op端口(biu_instr和op相连),为了证实所以有了测试3

     

     3f居然进了decode模块当操作码(op)处理了,它应该是一个立即数才对啊、再进入decode模块证实是否真的当操作码处理了,当我进入decode模块的时候发现真的悲剧了,没有使能信号控制,3f直接进入了操作码的case语句。

    初步认定核有问题

    在我写这个博客的时候觉得还有一丝希望的是,这个3f虽然当做操作码处理了,loop里面的所有数据都当操作码处理了,但是没有初始化完成,最后有一个使能把执行结果屏蔽了,—— 后待验证

  • 相关阅读:
    Node.js:工具模块
    Node.js:GET/POST请求
    shell之任务控制
    java实现杨辉三角系数
    java实现显示为树形
    java实现显示为树形
    java实现显示为树形
    java实现显示为树形
    java实现显示为树形
    java实现文件管理
  • 原文地址:https://www.cnblogs.com/sepeng/p/4137405.html
Copyright © 2011-2022 走看看