Prj10--8088单板机C语言8259中断测试(2)
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, addrmov al, dataout dx, al}}
char inp(unsigned int addr)
// 从I/O端口输入一字节{ char result;__asm{ mov dx, addrin al, dxmov result, al}return result;}void register_read(void){__asm{mov ax,CSmov cs_adr,axmov ax,DSmov ds_adr,axmov ax,SSmov 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 esxor ax, axmov es, axmov al, int_noxor ah, ahshl ax, 1shl ax, 1mov si, axmov ax, service_procmov es:[si], axinc siinc simov bx, csmov es:[si], bxpop 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, // 00x06, // 10x5B, // 20x4F, // 30x66, // 40x6D, // 50x7D, // 60x07, // 70x7F, // 80x6F // 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: 边沿触发 | 单片 | 需要ICW4outp(PIC1_CMD, 0x17);// ICW2: 中断向量基址=20houtp(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 clitiny_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 cliinit_8259();asm nopasm 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: 边沿触发 | 单片 | 需要ICW4outp(PIC1_CMD, 0x17);// ICW2: 中断向量基址=20houtp(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 cliinit_8259();asm nopasm sti
4.心得体会
这个程序调试的并不顺利,前一个晚上到10点,还没有头绪。主要犯了2个错误,1个是8259 的地址弄错了,应该是0x400,0x402;我写成了0x400,和0x401,程序下载,就跑飞。2是8259的ICW2写的中断向量写成了0x02,我想偷懒,想借用前面NMI中断响应函数,结果查DeepSeeK,ICW2应该是8的整数倍,写0x02,默认为0号中断,可0号中断我没有放中断响应函数,所以跑飞了。
今早一起来,就赶紧跑来把想的修改测试了,结果一下就跑通了。
写程序,严重影响睡眠,这个问题闹得我昨晚一宿都没咋睡好。因为这是10个例子程序的最后一个,我有迫切想把这个计划上句号。
万幸,总算过去了。
相关文章:

Prj10--8088单板机C语言8259中断测试(2)
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[]"Hel…...
《前端面试题:CSS对浏览器兼容性》
CSS浏览器兼容性完全指南:从原理到实战 跨浏览器兼容性是前端开发的核心挑战,也是面试中的高频考点。查看所有css属性对各个浏览器兼容网站:https://caniuse.com 一、浏览器兼容性为何如此重要? 在当今多浏览器生态中,…...
使用 Docker Compose 安装 Redis 7.2.4
前面是指南,后面是主要步骤实际执行日志 使用 Docker Compose 安装 Redis 7.2.4 以下是使用 Docker Compose 安装 Redis 7.2.4 的完整指南: 1. 创建项目目录和文件 bash 复制 下载 # 创建项目目录 mkdir redis-docker && cd redis-docker#…...

35.x64汇编写法(二)
免责声明:内容仅供学习参考,请合法利用知识,禁止进行违法犯罪活动! 本次游戏没法给 内容参考于:微尘网络安全 上一个内容:34.x64汇编写法(一) 上一个内容写了,汇编调…...

安全大模型的思考
马上要准备2025年的护网了,最近就一直很忙,被事情裹挟着前进,忙的晕头转向,近乎感冒,昨天部门搞了一场AI大模型培训,演讲者有着很深的技术底蕴,我听到了一句关于Sass数据验证这块大为感悟&#…...
SQL Server 2025 预览版新功能
T-SQL 语言增强 正则表达式 (Regex) 支持 功能概述: SQL Server 2025 在 T-SQL 中原生引入了 POSIX 兼容的正则表达式支持,通过内置函数(如 REGEXP_LIKE、REGEXP_REPLACE 等)可直接在查询中对文本进行复杂模式匹配、查找和替换。…...

NineData云原生智能数据管理平台新功能发布|2025年5月版
本月发布 6 项更新,其中重点发布 3 项、功能优化 3 项。 重点发布 数据库 DevOps - 多源敏感数据保护 敏感数据扫描能力大幅扩展,新增支持 TiDB、Doris、SelectDB、OceanBase MySQL、GreatSQL、StarRocks、ClickHouse、SingleStore、Lindorm 9 种大数据…...
数学复习笔记 25
今天能把第五章学完。加油。今年是最好上岸的一年。 5.23:全是单根,笑死,居然难受了。我现在每个题,都要总结。总结。总结实际上也总结不出啥东西。但是我一定要总结。主动让自己思考一下。老师的思路很清奇。他认为考的稀松平常…...

Linux可执行文件ELF文件结构
目标文件格式 编译器编译源代码后生成的文件叫做目标文件,而目标文件经过编译器链接之后得到的就是可执行文件。那么目标文件到底是什么?它和可执行文件又有什么区别?链接到底又做了什么呢?接下来,我们将探索一下目标…...
RAG:大模型微调的革命性增强——检索增强生成技术深度解析
RAG:大模型微调的革命性增强——检索增强生成技术深度解析 当大模型遇到知识瓶颈,RAG(检索增强生成)为模型装上"外部记忆库",让静态知识库与动态生成能力完美融合。本文将深入拆解RAG的技术原理、微调策略及…...

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

I2C通信讲解
I2C总线发展史 怎么在一条串口线上连接多个设备呢? 由于速度同步线是由主机实时发出的,所以主机可以按需求修改通信速度,这样在一条线上可以挂接不同速度的器件,单片机和性能差的器件通信,就输出较慢的脉冲信号&#x…...
【信息系统项目管理师-选择真题】2025上半年(第一批)综合知识答案和详解
更多内容请见: 备考信息系统项目管理师-专栏介绍和目录 文章目录 【第1题】【第2题】【第3题】【第4题】【第5题】【第6题】【第7题】【第8题】【第9题】【第10题】【第11题】【第12题】【第13题】【第14题】【第15题】【第16题】【第17题】【第18题】【第19题】【第20题】【第…...
ABP VNext 在 Kubernetes 中的零停机蓝绿发布
ABP VNext 在 Kubernetes 中的零停机蓝绿发布 🚀 📚 目录 ABP VNext 在 Kubernetes 中的零停机蓝绿发布 🚀📌 一、前提准备 ℹ️🧱 二、项目结构与目标 🎯🐳 三、多阶段 Dockerfile 构建 &#…...
linux 故障处置通用流程-36计-14-27
014:查看系统主要日志 查看以下日志: 主要查以下关键字 error/NIC/fs /"link down"/Oout of memory" /var/log/messages /var/log/dmesg 015:主机通讯是否延迟 执行命令: #ping 网关_IP #ping 关联主机_IP …...
https和http有什么区别-http各个版本有什么区别
http和 https的区别 HTTP(超文本传输协议)和 HTTPS(安全超文本传输协议)是两种用于在网络上传输数据的协议,它们的主要区别在于安全性: HTTP(Hypertext Transfer Protocol)&#x…...

基于回归算法的心理健康预测(EDA + 预测)
心理健康涵盖情感、心理与社会福祉,影响认知、情绪和行为模式,决定压力应对、人际交往及健康决策,且在生命各阶段(从童年至成年)均至关重要。心理健康与身体健康同为整体健康的核心要素:抑郁会增加糖尿病、…...
React Native开发鸿蒙运动健康类应用的项目实践记录
项目名称:HarmonyFitness - 基于React Native的鸿蒙运动健康应用 技术栈:React Native 0.72.5 TypeScript HarmonyOS API ArkTS原生模块 一、环境搭建与项目初始化 双环境配置 React Native环境: npx re…...

【新品解读】一板多能,AXRF49 定义新一代 RFSoC FPGA 开发平台
“硬件系统庞杂、调试周期长” “高频模拟前端不稳定,影响采样精度” “接收和发射链路难以同步,难以扩展更多通道” “数据流量大,处理与存储跟不上” 这些是大部分客户在构建多通道、高频宽的射频采样链路时,面临的主要问题。…...

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

YOLO在C#中的完整训练、验证与部署方案
YOLO在C#中的完整训练、验证与部署方案 C# 在 YOLO 部署上优势明显(高性能、易集成),但训练能力较弱,通常需结合 Python 实现。若项目对开发效率要求高且不依赖 C# 生态,建议全程使用 Python;若需深度集成…...
洛谷题目:P2761 软件补丁问题 (本题简单)
个人介绍: 题目传送门: P2761 软件补丁问题 - 洛谷 (luogu.com.cn) 前言: 这道题是一个典型的状态搜索问题,核心目标就是利用给定d额多个补丁程序,将包含若干错误的软件修复成没有错误的状态,并且要使得修复过程当中的总耗时最少。下面是小亦为大家阐述滴思路: 1、状态…...

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

【OpenGL学习】(四)统一着色和插值着色
文章目录 【OpenGL学习】(四)统一着色和插值着色统一着色(Flat/Uniform Shading)插值着色(Interpolated Shading) 【OpenGL学习】(四)统一着色和插值着色 着色器介绍: h…...
42、响应处理-【源码分析】-浏览器与PostMan内容协商完全适配
42、响应处理源码分析浏览器与PostMan内容协商完全适配 要实现浏览器与PostMan在内容协商上的完全适配,需要在Spring Boot应用中自定义内容协商策略,确保服务器能根据浏览器和PostMan的请求头正确返回合适格式的数据。以下是详细的步骤: ### …...

在 CentOS 上安装 Docker 和 Docker Compose 并配置使用国内镜像源
在 CentOS 上安装 Docker 和 Docker Compose 并配置使用国内镜像源,可以加速镜像下载速度。以下是详细的步骤: 一、安装 Docker 移除旧版本的 Docker(如果有): sudo yum remove docker \docker-client \docker-client…...
Java Lambda表达式深度解析:从入门到实战
简介 Lambda表达式是Java 8引入的最重要特性之一,它极大地简化了Java代码的编写方式,使函数式编程风格在Java中成为可能。本文将全面介绍Lambda表达式的概念、语法、应用场景以及与相关特性的配合使用,帮助开发者掌握这一强大的编程工具。 一、Lambda表达式基础 1.1 什么…...

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

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