Prj08--8088单板机C语言8255读取按键码

news2025/6/7 13:33:14

1.验证结果

2.代码片

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

3.完整代码

 

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

//


//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);
     // 使用安全格式化
    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);
    init_8255();
   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,"NMI Interrupt count=%x    \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);


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




4.项目进度

 

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

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

相关文章

蜜獾算法(HBA,Honey Badger Algorithm)

2021年由Hashim等人提出&#xff08;论文&#xff1a;Honey Badger Algorithm: A New Metaheuristic Algorithm for Solving Optimization Problems&#xff09;。模拟蜜獾在自然界中的智能捕食行为&#xff0c;属于群体智能优化算法&#xff08;与粒子群PSO、遗传算法GA同属一…

Modbus转Ethernet IP网关助力罗克韦尔PLC数据交互

在工业自动化领域&#xff0c;Modbus协议是一种广泛应用的串行通信协议&#xff0c;它定义了主站和从站之间的通信规则和数据格式。罗克韦尔PLC是一种可编程的逻辑控制器&#xff0c;通过Modbus协议实现与其他设备之间的数据交互。然而&#xff0c;随着以太网技术的普及和发展&…

飞算JavaAI 炫技赛重磅回归!用智能编码攻克老项目重构难题

深夜还在排查十年前Hibernate框架埋下的N1查询隐患&#xff1f;跨语言迁移时发现SpringMVC控制器里的业务逻辑像一团乱麻&#xff1f;当企业数字化进入深水区&#xff0c;百万行代码的老系统就像一座随时可能崩塌的"技术债冰山"。近日&#xff0c;飞算科技发布JavaAI…

ToolsSet之:XML工具

ToolsSet是微软商店中的一款包含数十种实用工具数百种细分功能的工具集合应用&#xff0c;应用基本功能介绍可以查看以下文章&#xff1a; Windows应用ToolsSet介绍https://blog.csdn.net/BinField/article/details/145898264 ToolsSet中Text菜单下的XML Tool工具是一个Xml工…

keepalived定制日志bug

keepalived定制日志bug 源码安装apt安装endl 源码安装 在/etc/rsyslog.d/目录下创建 keepalived的日志配置文件keepalived.conf [rootubuntu24-13:~]# vim /etc/rsyslog.d/keepalived.conf [rootubuntu24-13:~]# cat /etc/rsyslog.d/keepalived.conf local6.* /var/log/keepa…

数据库系统概论(十三)详细讲解SQL中数据更新(插入,修改与更新)

数据库系统概论&#xff08;十三&#xff09;详细讲解SQL中数据更新 前言一、数据插入1. 插入数据是什么&#xff1f;2.插入单条数据&#xff08;插入元组&#xff09;场景 1&#xff1a;指定部分列插入场景 2&#xff1a;不指定列名&#xff08;插入所有列&#xff09;场景 3&…

极客时间-《搞定音频技术》-学习笔记

极客时间-《搞定音频技术》-学习笔记 语音基础知识 https://www.zhangzhenhu.com/audio/feature.html 序章-0 作者说这个语音技术啊&#xff0c;未来肯定前景大好啊&#xff0c;大家都来学习&#xff0c;然后给出了课程的脑图 音频基础 什么是声音 声音的三要素是指响度、…

网络攻防技术十三:网络防火墙

文章目录 一、网络防火墙概述1、网络型防火墙&#xff08;网络防火墙&#xff09;2、Web应用防火墙3、数据库防火墙4、主机防火墙&#xff08;个人防火墙&#xff09;5、网络防火墙的功能 二、防火墙工作原理1、无状态包过滤防火墙2、有状态包过滤防火墙&#xff08;状态检测/动…

Express 集成Sequelize+Sqlite3 默认开启WAL 进程间通信 Conf 打包成可执行 exe 文件

代码&#xff1a;express-exe: 将Express开发的js打包成exe服务丢给客户端使用 实现目标 Express 集成 Sequelize 操作 Sqlite3 数据库&#xff1b; 启动 Sqlite3 时默认开启 WAL 模式&#xff0c;避免读写互锁&#xff0c;支持并发读&#xff1b; 利用 Conf 实现主进程与 Ex…

2024年认证杯SPSSPRO杯数学建模D题(第二阶段)AI绘画带来的挑战解题全过程文档及程序

2024年认证杯SPSSPRO杯数学建模 D题 AI绘画带来的挑战 原题再现&#xff1a; 2023 年开年&#xff0c;ChatGPT 作为一款聊天型AI工具&#xff0c;成为了超越疫情的热门词条&#xff1b;而在AI的另一个分支——绘图领域&#xff0c;一款名为Midjourney&#xff08;MJ&#xff…

DOCKER使用记录

1、拉取镜像 直接使用docker pull <image>&#xff0c;大概率会出现下面的报错信息&#xff1a; (base) jetsonyahboom:~$ docker pull ubuntu:18.04 Error response from daemon: Get "https://registry-1.docker.io/v2/": net/http: request canceled while …

【深度学习相关安装及配环境】Anaconda搭建虚拟环境并安装CUDA、cuDVV和对应版本的Pytorch,并在jupyter notebook上部署

目录 1. 查看自己电脑的cuda版本2.安装cuda关于环境变量的配置测试一下&#xff0c;安装完成 3.安装cuDVV环境变量的配置测试一下&#xff0c;安装完成 4.创建虚拟环境先安装镜像源下载3.11版本py 5.在虚拟环境下&#xff0c;下载pytorch6.验证是否安装成功7.在jupyter noteboo…

web3-区块链基础:从区块添加机制到哈希加密与默克尔树结构

区块链基础&#xff1a;从区块添加机制到哈希加密与默克尔树结构 什么是区块链 抽象的回答: 区块链提供了一种让多个参与方在没有一个唯一可信方的情况下达成合作 若有可信第三方 > 不需要区块链 [金融系统中常常没有可信的参与方] 像股票市场&#xff0c;或者一个国家的…

TCP小结

1. 核心特性 面向连接&#xff1a;通过三次握手建立连接&#xff0c;四次挥手终止连接&#xff0c;确保通信双方状态同步。 TCP连接建立的3次握手 抓包&#xff1a; client发出连接请求&#xff1b; server回应client请求&#xff0c;并且同步发送syn连接&#xff1b; clien…

Python 打包指南:setup.py 与 pyproject.toml 的全面对比与实战

在 Python 开发中&#xff0c;创建可安装的包是分享代码的重要方式。本文将深入解析两种主流打包方法——setup.py 和 pyproject.toml&#xff0c;并通过一个实际项目示例&#xff0c;展示如何使用现代的 pyproject.toml 方法构建、测试和发布 Python 包。 一、setup.py 与 pyp…

性能优化 - 案例篇:缓存_Guava#LoadingCache设计

文章目录 Pre引言1. 缓存基本概念2. Guava 的 LoadingCache2.1 引入依赖与初始化2.2 手动 put 与自动加载&#xff08;CacheLoader&#xff09;2.2.1 示例代码 2.3 缓存移除与监听&#xff08;invalidate removalListener&#xff09; 3. 缓存回收策略3.1 基于容量的回收&…

python入门(1)

第一章 第一个python程序 1.1 print函数 print方法的作用 : 把想要输出的内容打印在屏幕上 print("Hello World") 1.2 输出中文 在Python 2.x版本中&#xff0c;默认的编码方式是ASCII编码方式&#xff0c;如果程序中用到了中文&#xff0c;直接输出结果很可能会…

【PDF提取表格】如何提取发票内容文字并导出到Excel表格,并将发票用发票号改名,基于pdf电子发票的应用实现

应用场景 该应用主要用于企业财务部门或个人处理大量电子发票&#xff0c;实现以下功能&#xff1a; 自动从 PDF 电子发票中提取关键信息&#xff08;如发票号码、日期、金额、销售方等&#xff09;将提取的信息整理并导出到 Excel 表格&#xff0c;方便进行财务统计和报销使…

Hugging Face 最新开源 SmolVLA 小模型入门教程(一)

系列文章目录 目录 系列文章目录 前言 一、引言 二、认识 SmolVLA&#xff01; 三、如何使用SmolVLA&#xff1f; 3.1 安装 3.2 微调预训练模型 3.3 从头开始训练 四、方法 五、主要架构 5.1 视觉语言模型&#xff08;VLM&#xff09; 5.2 动作专家&#xff1a;流匹…

封闭内网安装配置VSCode Anconda3 并配置 PyQt5开发

封闭内网安装配置VSCode Anconda3 并配置 PyQt5开发 零一 vscode1.1 下载 vscode1.2 下载插件1.3 安装 二 anaconda 32.1 下载2.2 新建虚拟环境1 新建快捷方式,启动base2 新建虚拟环境 3 配置Qt designer3.1 designer.exe和uic.exe3.2 设置插件,3.4 ui文件转为py文件 4使用4.1 …