Prj10--8088单板机C语言8259中断测试(2)

news2025/6/7 17:38:59

1.测试结果

2.全部代码

#include "tiny_stdarg.h"  // 使用自定义可变参数实现

#define ADR_273 0x0200
#define ADR_244 0x0400
#define  LED_PORT   0x800
#define  PC16550_THR   0x1f0
#define  PC16550_LSR   0x1f5
/
//基本的IO操作函数
/
char str[]="Hello World!  20250531  Very Ok!!!\r\n";
//char  buff[60]
char cx='A';
unsigned int cs_adr=0,ds_adr=0,ss_adr=0;

/// @brief 
/// @param addr 
/// @param data 
void outp(unsigned int addr, char data)
// 输出一字节到I/O端口
 { __asm
    { mov dx, addr
      mov al, data
      out dx, al
    }
 }
char inp(unsigned int addr)
// 从I/O端口输入一字节
 { char result;
   __asm
    { mov dx, addr
      in al, dx
      mov result, al
    }
   return result;
 }
 void register_read(void)
 {
    __asm
    {
        mov ax,CS
        mov cs_adr,ax
        mov ax,DS
        mov ds_adr,ax
        mov ax,SS
        mov ss_adr,ax
    }

 }

//串口发送函数

void  uart_send(char x)
{
 int temp;
  while(1)
  {
   temp=inp(PC16550_LSR);
   if((temp&0x20)==0x20)
   {
    break;
    }
   }
  outp(PC16550_THR,x);
 }

void uart_str_send(char *p)
{
 //int i=0;
 //char str1[20]="Hello World!\r\n";
 //char *p;
 //p=str1;
 
 while(*p!='\0')
  {
    uart_send(*p);
    p++;
   }

/*
 for(i=0;i<14;i++)
   {
     uart_send(str1[i]);
    }
*/
 }
///
/* sprintf()函数实现 */
/* tiny_sprintf.c */
#include "tiny_stdarg.h"

static void itoa(unsigned num, int base, char *out) {
    char buf[6]; // 16位整数最大5位数字 + 结束符
    char *p = buf;
    int i = 0;
    
    if (num == 0) {
        *out++ = '0';
        *out = '\0';
        return;
    }
    
    while (num > 0) {
        int r = num % base;
        *p++ = (r < 10) ? (r + '0') : (r - 10 + 'a');
        num /= base;
        i++;
    }
    
    while (i-- > 0) {
        *out++ = *--p;
    }
    *out = '\0';
}

int tiny_sprintf(char *buf, const char *fmt, ...) {
    va_list args;
    char *p = buf;
    const char *s = fmt;
    
    va_start(args, fmt);
    
    while (*s) {
        if (*s != '%') {
            *p++ = *s++;
            continue;
        }
        
        s++;
        switch (*s) {
            case 'd': {
                int num = va_arg(args, int);
                if (num < 0) {
                    *p++ = '-';
                    num = -num;
                }
                itoa(num, 10, p);
                while (*p) p++;
                s++;
                break;
            }
            case 'x': {
                unsigned num = va_arg(args, unsigned);
                itoa(num, 16, p);
                while (*p) p++;
                s++;
                break;
            }
            case 's': {
                char *str = va_arg(args, char *);
                while (*str) *p++ = *str++;
                s++;
                break;
            }
            case 'c': {
                char c = (char)va_arg(args, int);
                *p++ = c;
                s++;
                break;
            }
            case '%': {
                *p++ = '%';
                s++;
                break;
            }
            default: {
                *p++ = '%';
                *p++ = *s++;
                break;
            }
        }
    }
    
    *p = '\0';
    va_end(args);
    return p - buf;
}
///
//NMI 中断
//
/* NMI 计数器 */
volatile unsigned char nmi_count =10;
//设置中断失量表 
void set_int(unsigned char int_no, void * service_proc)
 { _asm
    { push es
      xor ax, ax
      mov es, ax
      mov al, int_no
      xor ah, ah
      shl ax, 1
      shl ax, 1
      mov si, ax
      mov ax, service_proc
      mov es:[si], ax
      inc si
      inc si
      mov bx, cs
      mov es:[si], bx
      pop es
    }
 }
//中断处理函数
/*
void _interrupt  near nmi_handler(void)
 {
  	nmi_count++;
 }
    */
//
//8255  
//
// 定义8255端口地址 (根据原理图译码确定)
#define PORT_8255_A 0x200  // PA端口地址
#define PORT_8255_B 0x201  // PB端口地址
#define PORT_8255_C 0x202  // PC端口地址
#define PORT_8255_CTRL 0x203 // 控制寄存器地址

// 数码管段码表 (共阴极)
unsigned char seg_codes[] = {
    0x3F, // 0
    0x06, // 1
    0x5B, // 2
    0x4F, // 3
    0x66, // 4
    0x6D, // 5
    0x7D, // 6
    0x07, // 7
    0x7F, // 8
    0x6F  // 9
};

// 延时函数
void delay(unsigned int ms) {
    for (unsigned int i = 0; i < ms; i++) {
        for (unsigned int j = 0; j < 100; j++) {
            // 空循环延时
        }
    }
}

// 初始化8255
void init_8255() {
    // 控制字: 10000001 (0x81)
    // A口输出, B口输出, C口输出
    outp(PORT_8255_CTRL, 0x81);
}

// 显示8位数字
void display_numbers() {
    unsigned char digits[] = {1, 2, 3, 4, 5, 6, 7, 8}; // 要显示的数字
    
    while (1) {  // 按任意键退出
        for (int i = 0; i < 8; i++) {
            // 设置位选 (选中当前位)
            outp(PORT_8255_B, ~(1 << i));
            
            // 设置段码
            outp(PORT_8255_A, ~seg_codes[digits[i]]);
            
            // 延时保持显示
            delay(1);
            
            // 关闭当前位显示 (消除鬼影)
            outp(PORT_8255_A, 0x00);
        }
    }
}

//
//8253 part
//
// 8253定时器端口定义
#define PORT_8253_CNT0 0x300      // 计数器0地址
#define PORT_8253_CNT1 0x301      // 计数器1地址
#define PORT_8253_CNT2 0x302      // 计数器2地址
#define PORT_8253_CTRL 0x303      // 控制寄存器地址

// 时钟频率定义 (根据原理图第4页)
#define PCLK_FREQUENCY 1193182     // 标准8253时钟频率(1.193182MHz)
#define OUTPUT_FREQUENCY 200      // 目标输出频率(1kHz)

// 计算计数器初值
#define COUNTER_VALUE (unsigned int)(PCLK_FREQUENCY / OUTPUT_FREQUENCY)

// 初始化8253定时器
void init_8253() {
    // 控制字: 00110110 (0x36)
    // 选择计数器0 | 写入高低字节 | 模式3(方波) | 二进制计数
    outp(PORT_8253_CTRL, 0x36);
    
    // 写入计数器初值 (先低字节后高字节)
    outp(PORT_8253_CNT0, COUNTER_VALUE & 0xFF);         // 低字节
    outp(PORT_8253_CNT0, (COUNTER_VALUE >> 8) & 0xFF);  // 高字节
}

//
//8259 part
//
#define PIC1_CMD  0x400   // 命令端口 (A0=0)
#define PIC1_DATA 0x402  // 数据端口 (A0=1)  !!!!!!!!!!

// 初始化8259(设置自动EOI)
void init_8259(void) {
    // ICW1: 边沿触发 | 单片 | 需要ICW4
    outp(PIC1_CMD, 0x17);
    
    // ICW2: 中断向量基址=20h
    outp(PIC1_DATA, 0x08);
    
    // ICW4: 8086模式 | 自动EOI (0x03)
    outp(PIC1_DATA, 0x0f);  // 关键修改:设置自动EOI模式
    
    // OCW1: 只允许IR0中断 (11111110b)
    outp(PIC1_DATA, 0xfe);
}

//


//char  end_flag[5]={0x55,0x55,0x55,0x55,0x55};
extern void nmi_handler(void);
void main(void)
/*检测按键状态并由LED发光二极管显示,
  若按键闭合对应LED发光二极管点亮,
  若按键断开对应LED发光二极管灭.*/
 { 
   int i=0;
   char buffer[80];
   unsigned char key_code=0xff;   // 使用安全格式化
    //tiny_sprintf(buffer, "Hex: %x\n",255);
     // 使用安全格式化
     asm cli
    tiny_sprintf(buffer, 
                "Decimal: %d    \n"
                "Hex: %x    \n"
                "String: %s    \r\n", 
                -123, 
                0xABCD, 
                "Hello");
    register_read();        
    //set_nmi_handler(); 
    set_int(0x02, (void *)&nmi_handler);
    set_int(0x08, (void *)&nmi_handler);//8259   与NMI共用一个中断服务函数
   
    init_8255();
    init_8253();
  
    asm  cli
    init_8259();
    asm nop
    asm  sti 

   while (1)
   {
     //char button_state;
     //button_state=inp(ADR_244);
     //int i=0;
     //uart_str_send(str);
     uart_str_send(buffer);

     tiny_sprintf(buffer,"******************************************\r\n");
     uart_str_send(buffer);
     tiny_sprintf(buffer,"CS_ADR= 0X%x     \r\n",cs_adr);
     uart_str_send(buffer);
      tiny_sprintf(buffer,"DS_ADR= 0X%x     \r\n",ds_adr);
     uart_str_send(buffer);
     tiny_sprintf(buffer,"SS_ADR= 0X%x     \r\n",ss_adr);
     uart_str_send(buffer);

     tiny_sprintf(buffer,"8259 Interrupt count=%d    \r\n",nmi_count);
     uart_str_send(buffer);
     tiny_sprintf(buffer,"******************************************\r\n");
     uart_str_send(buffer);

     key_code=inp(PORT_8255_C)&0x0f;
     tiny_sprintf(buffer,"Key_code= 0X%x     \r\n",key_code);
     uart_str_send(buffer);

     //asm  int 8
     //uart_send(cx);

     for(i=0;i<5000;i++);
     for(i=0;i<5000;i++);
     outp(LED_PORT, 0xff);
     for(i=0;i<5000;i++);
     for(i=0;i<5000;i++);
     outp(LED_PORT, 0x00);

     //display_numbers();
   }
 }
char  end_flag[5]={0x55,0x55,0x55,0x55,0x55};




3.核心代码片

//
//8259 part
//
#define PIC1_CMD  0x400   // 命令端口 (A0=0)
#define PIC1_DATA 0x402  // 数据端口 (A0=1)  !!!!!!!!!!

// 初始化8259(设置自动EOI)
void init_8259(void) {
    // ICW1: 边沿触发 | 单片 | 需要ICW4
    outp(PIC1_CMD, 0x17);
    
    // ICW2: 中断向量基址=20h
    outp(PIC1_DATA, 0x08);
    
    // ICW4: 8086模式 | 自动EOI (0x03)
    outp(PIC1_DATA, 0x0f);  // 关键修改:设置自动EOI模式
    
    // OCW1: 只允许IR0中断 (11111110b)
    outp(PIC1_DATA, 0xfe);
}

 

//set_nmi_handler(); 
    set_int(0x02, (void *)&nmi_handler);
    set_int(0x08, (void *)&nmi_handler);//8259   与NMI共用一个中断服务函数
   
    init_8255();
    init_8253();
  
    asm  cli
    init_8259();
    asm nop
    asm  sti 

 4.心得体会

这个程序调试的并不顺利,前一个晚上到10点,还没有头绪。主要犯了2个错误,1个是8259 的地址弄错了,应该是0x400,0x402;我写成了0x400,和0x401,程序下载,就跑飞。2是8259的ICW2写的中断向量写成了0x02,我想偷懒,想借用前面NMI中断响应函数,结果查DeepSeeK,ICW2应该是8的整数倍,写0x02,默认为0号中断,可0号中断我没有放中断响应函数,所以跑飞了。

今早一起来,就赶紧跑来把想的修改测试了,结果一下就跑通了。

写程序,严重影响睡眠,这个问题闹得我昨晚一宿都没咋睡好。因为这是10个例子程序的最后一个,我有迫切想把这个计划上句号。

万幸,总算过去了。

 

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/2401854.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

35.x64汇编写法(二)

免责声明&#xff1a;内容仅供学习参考&#xff0c;请合法利用知识&#xff0c;禁止进行违法犯罪活动&#xff01; 本次游戏没法给 内容参考于&#xff1a;微尘网络安全 上一个内容&#xff1a;34.x64汇编写法&#xff08;一&#xff09; 上一个内容写了&#xff0c;汇编调…

安全大模型的思考

马上要准备2025年的护网了&#xff0c;最近就一直很忙&#xff0c;被事情裹挟着前进&#xff0c;忙的晕头转向&#xff0c;近乎感冒&#xff0c;昨天部门搞了一场AI大模型培训&#xff0c;演讲者有着很深的技术底蕴&#xff0c;我听到了一句关于Sass数据验证这块大为感悟&#…

NineData云原生智能数据管理平台新功能发布|2025年5月版

本月发布 6 项更新&#xff0c;其中重点发布 3 项、功能优化 3 项。 重点发布 数据库 DevOps - 多源敏感数据保护 敏感数据扫描能力大幅扩展&#xff0c;新增支持 TiDB、Doris、SelectDB、OceanBase MySQL、GreatSQL、StarRocks、ClickHouse、SingleStore、Lindorm 9 种大数据…

Linux可执行文件ELF文件结构

目标文件格式 编译器编译源代码后生成的文件叫做目标文件&#xff0c;而目标文件经过编译器链接之后得到的就是可执行文件。那么目标文件到底是什么&#xff1f;它和可执行文件又有什么区别&#xff1f;链接到底又做了什么呢&#xff1f;接下来&#xff0c;我们将探索一下目标…

DisplayPort 2.0协议介绍(1)

最近开始学习DisplayPort 2.0协议&#xff0c;相比于DP1.4a&#xff0c;最主要的是速率提升到了10Gbps/lane&#xff0c;还有就是128b/132b编码方式的修改。至于速率13.5Gbps和20Gbps还只是可选项&#xff0c;在DP2.1协议才成为必须支持选项。 那在实现技术细节上有哪些变化呢…

I2C通信讲解

I2C总线发展史 怎么在一条串口线上连接多个设备呢&#xff1f; 由于速度同步线是由主机实时发出的&#xff0c;所以主机可以按需求修改通信速度&#xff0c;这样在一条线上可以挂接不同速度的器件&#xff0c;单片机和性能差的器件通信&#xff0c;就输出较慢的脉冲信号&#x…

基于回归算法的心理健康预测(EDA + 预测)

心理健康涵盖情感、心理与社会福祉&#xff0c;影响认知、情绪和行为模式&#xff0c;决定压力应对、人际交往及健康决策&#xff0c;且在生命各阶段&#xff08;从童年至成年&#xff09;均至关重要。心理健康与身体健康同为整体健康的核心要素&#xff1a;抑郁会增加糖尿病、…

【新品解读】一板多能,AXRF49 定义新一代 RFSoC FPGA 开发平台

“硬件系统庞杂、调试周期长” “高频模拟前端不稳定&#xff0c;影响采样精度” “接收和发射链路难以同步&#xff0c;难以扩展更多通道” “数据流量大&#xff0c;处理与存储跟不上” 这些是大部分客户在构建多通道、高频宽的射频采样链路时&#xff0c;面临的主要问题。…

贪心算法应用:线性规划贪心舍入问题详解

贪心算法应用&#xff1a;线性规划贪心舍入问题详解 贪心算法是一种在每一步选择中都采取当前状态下最优的选择&#xff0c;从而希望导致结果是全局最优的算法策略。在线性规划问题中&#xff0c;贪心算法特别是贪心舍入技术有着广泛的应用。下面我将全面详细地讲解这一主题。…

YOLO在C#中的完整训练、验证与部署方案

YOLO在C#中的完整训练、验证与部署方案 C# 在 YOLO 部署上优势明显&#xff08;高性能、易集成&#xff09;&#xff0c;但训练能力较弱&#xff0c;通常需结合 Python 实现。若项目对开发效率要求高且不依赖 C# 生态&#xff0c;建议全程使用 Python&#xff1b;若需深度集成…

洛谷题目:P2761 软件补丁问题 (本题简单)

个人介绍: 题目传送门: P2761 软件补丁问题 - 洛谷 (luogu.com.cn) 前言: 这道题是一个典型的状态搜索问题,核心目标就是利用给定d额多个补丁程序,将包含若干错误的软件修复成没有错误的状态,并且要使得修复过程当中的总耗时最少。下面是小亦为大家阐述滴思路: 1、状态…

智慧园区数字孪生全链交付方案:降本增效30%,多案例实践驱动全周期交付

在智慧园区建设浪潮中&#xff0c;数字孪生技术正成为破解传统园区管理难题的核心引擎。通过构建与物理园区1:1映射的数字模型&#xff0c;实现数据集成、状态同步与智能决策&#xff0c;智慧园区数字孪生全链交付方案已在多个项目中验证其降本增效价值——某物流园区通过该方案…

【OpenGL学习】(四)统一着色和插值着色

文章目录 【OpenGL学习】&#xff08;四&#xff09;统一着色和插值着色统一着色&#xff08;Flat/Uniform Shading&#xff09;插值着色&#xff08;Interpolated Shading&#xff09; 【OpenGL学习】&#xff08;四&#xff09;统一着色和插值着色 着色器介绍&#xff1a; h…

在 CentOS 上安装 Docker 和 Docker Compose 并配置使用国内镜像源

在 CentOS 上安装 Docker 和 Docker Compose 并配置使用国内镜像源&#xff0c;可以加速镜像下载速度。以下是详细的步骤&#xff1a; 一、安装 Docker 移除旧版本的 Docker&#xff08;如果有&#xff09;&#xff1a; sudo yum remove docker \docker-client \docker-client…

Docker慢慢学

1、Docker DeskTop 2、N8N下载 docker run -p 8888:5678 n8nio/n8n 3、Kafka kafka依赖zookeeper,先启动zookeeper docker pull zookeeper docker run -d --name zookeeper -p 2181:2181 -e ALLOW_ANONYMOUS_LOGINyes zookeeper 启动kafka docker pull confluentinc/cp…

cursor-free-vip使用

一、项目简介 Cursor-Free-VIP 是一个开源项目&#xff0c;旨在帮助用户免费使用 Cursor AI 的高级功能。它通过自动注册 Cursor 账号、重置机器 ID 和完成 Auth 验证等操作&#xff0c;解决 Cursor AI 中常见的限制提示。 二、系统准备 1…cursor需要更新到最新的版本 三、…

使用SSH tunnel访问内网的MySQL

文章目录 环境背景方法参考 注&#xff1a;本文是使用SSH tunnel做端口转发的一个示例。有关SSH端口转发&#xff0c;可参考我的几篇文档 https://blog.csdn.net/duke_ding2/article/details/106878081https://blog.csdn.net/duke_ding2/article/details/135627263https://blo…

Redis持久化模式RDB与AOF

RDB持久化 RDB也被叫做Redis数据快照。简单来说就是把内存中的所有数据记录到磁盘中。当Redis实例故障重启后重磁盘中读取快照文件进行数据恢复(快照文件默认保存在当前运行目录)&#xff1b; 演示Redis正常停机自动执行一次RDB操作 配置Redis触发RDB机制 RDB其它配置也可在red…

【数据结构】树形结构--二叉树(二)

【数据结构】树形结构--二叉树&#xff08;二&#xff09; 一.二叉树的实现1.求二叉树结点的个数2.求二叉树叶子结点的个数3.求二叉树第k层结点的个数4.求二叉树的深度&#xff08;高度&#xff09;5.在二叉树中查找值为x的结点6.判断二叉树是否为完全二叉树7.二叉树的销毁 一.…

深度学习题目1

梯度下降法的正确步骤是什么&#xff1f; a.计算预测值和真实值之间的误差 b.重复迭代&#xff0c;直至得到网络权重的最佳值 c.把输入传入网络&#xff0c;得到输出值 d.用随机值初始化权重和偏差 e.对每一个产生误差的神经元&#xff0c;调整相应的&#xff08;权重&#xff…