当前位置: 首页 > news >正文

Prj11-8088单板机C语言大综合(一)

【免费】8088单板机C语言大综合代码与编译器资源-CSDN下载

1.实现目标

   利用C语言实现8088单板机8259,8253,8255以及串口通讯PC16550的综合应用测试。利用8253产生定时脉冲,定时脉冲送给INTEL8259,产生定时中断,定时中断程序,动态刷新8255的数码管显示。

2.8253定时脉冲的实现

8253定时器硬件电路

配置代码

//////////////////////////////////////////////////////////////////////////////////
//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 OUTPUT_FREQUENCY 180     // 目标输出频率(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);  // 高字节
}

3.8259的定时中断实现

配置程序

//////////////////////////////////////////////////////////////////////////////////
//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);
}

中断响应函数地址设定

//设置中断失量表 
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}}.............
set_int(0x08, (void *)&nmi_handler);//8259   与NMI共用一个中断服务函数
.............
_nmi_handler PROC NEARpush ax                 ; 保存寄存器inc _nmi_count          ; nmi_count++ (C变量)pop ax                  ; 恢复寄存器call near ptr _display_intriret                    ; 中断返回
_nmi_handler ENDP

4.8255数码管显示

动态显示程序

// 初始化8255
void init_8255() {// 控制字: 10000001 (0x81)// A口输出, B口输出, C口输出outp(PORT_8255_CTRL, 0x81);
}//定时中断,8位数码管动态显示
//int ip_disp=0;
void display_intr()
{unsigned char digits[] = {1, 2, 3, 4, 5, 6, 7, 8}; // 要显示的数字static int ip_disp=0;// 设置位选 (选中当前位)outp(PORT_8255_B, ~(1 << ip_disp));// 设置段码outp(PORT_8255_A, ~seg_codes[digits[ip_disp]]);//outp(PORT_8255_A, 0x06);// 延时保持显示delay(1);// 关闭当前位显示 (消除鬼影)//outp(PORT_8255_A, 0x00);ip_disp++;if(ip_disp>7){ip_disp=0;}
}

5.完整的程序代码

1)汇编部分

.MODEL TINY
.8086.code
ORG 100h        ; COM 程序入口偏移PUBLIC _nmi_handler     ; 导出符号供C使用extrn _main:procextrn _display_intr:procstart:; 手动设置堆栈(Tiny 模式要求)mov ax, csmov ss, axmov sp, 05fffh;add ax,0100hmov ds,axmov es,ax; 调用 C 主函数call near ptr _main; 退出到 DOS(使用 AL 中的返回码)mov ah, 4Ch  ; DOS 功能:程序退出int 21h_nmi_handler PROC NEARpush ax                 ; 保存寄存器inc _nmi_count          ; nmi_count++ (C变量)pop ax                  ; 恢复寄存器call near ptr _display_intriret                    ; 中断返回
_nmi_handler ENDP.dataEXTERN _nmi_count:WORD  ; 声明外部C变量public __acrtused__acrtused = 9876h.stackEND start       ; 指定入口点为 start

2)C部分

#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);}}
}//定时中断,8位数码管动态显示
//int ip_disp=0;
void display_intr()
{unsigned char digits[] = {1, 2, 3, 4, 5, 6, 7, 8}; // 要显示的数字static int ip_disp=0;// 设置位选 (选中当前位)outp(PORT_8255_B, ~(1 << ip_disp));// 设置段码outp(PORT_8255_A, ~seg_codes[digits[ip_disp]]);//outp(PORT_8255_A, 0x06);// 延时保持显示delay(1);// 关闭当前位显示 (消除鬼影)//outp(PORT_8255_A, 0x00);ip_disp++;if(ip_disp>7){ip_disp=0;}
}//////////////////////////////////////////////////////////////////////////////////
//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 OUTPUT_FREQUENCY 180     // 目标输出频率(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);//display_intr();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};

6.编译

Build.bat编译批处理

..\tools\ml.exe /c /AT /Zm /Zi rtl.asm    
..\tools\dmc.exe -mt -0 -c -g  -ND  main.c      
..\tools\optlink.exe /TINY   /NODEF:SDS rtl.obj main.obj, debug.com  /MAP  /Fm  debug.map

7.下载测试

8.测试结果

测试通过!

http://www.dtcms.com/a/414981.html

相关文章:

  • 44.网络层
  • 肇庆网站制作软件郑州企业网络推广公司
  • ALLaM - 专为阿拉伯语设计的AI大语言模型
  • Docker Compose 停止命令对比
  • 北京网站推广优化更改wordpress端口
  • 优势的seo网站优化排名网站内容质量
  • Transformer 能做什么?—— 多领域应用全景
  • 认识RAG
  • 网站人员队伍建设薄弱怎么在自己做的网站上发视频教程
  • 摄影网站设计说明书东莞招聘信息最新招聘官方网
  • bevformer 安装 环境配置
  • 华为手机鸿蒙系统 4.2 / 4.3 安装谷歌框架的详细教程
  • 南昌网站开发爱网站长尾
  • 工业摄像头是应用于工业现场的高性能数字图像采集设备
  • 【08】VisionMaster入门到精通——卡尺工具和边缘查找
  • RRateLimiter的使用
  • 做网站html整合资源加强全市网站建设
  • 使用opencv来识别信用卡的号码
  • 【rabbitmq 高级特性】全面详解RabbitMQ重试机制
  • 在飞腾D2000/8平台下ubuntu内核添加WX1860和WX1820的驱动
  • docker相关进程的作用
  • 建设的网站如何让用户注册宁波发布最新通报
  • [LVGL] 中国象棋
  • 通过XShell使用Git三板斧
  • 【Git】远程操作 + 给命令配置别名 + 标签管理
  • 教学网站建设计划免费ppt下载网站
  • 给客户做网站需要提供看电视剧免费的网站
  • 分治法找到数组中出现次数超过一半的元素
  • C语言入门知识点(12.回调函数与qsort函数的模拟与实现)
  • 徐州网站客户如何做网站的seo优化