news 2026/5/11 5:02:15

27.MPSOC FPGA 驱动PCF8574

作者头像

张小明

前端开发工程师

1.2k 24
文章封面图
27.MPSOC FPGA 驱动PCF8574

上一篇记录了AHT20,本篇记录下PCF8574,原理类似:

#include <stdio.h> #include <stdlib.h> #include <fcntl.h> #include <unistd.h> #include <sys/ioctl.h> #include <linux/i2c-dev.h> #include <errno.h> #include <stdint.h> /************************** PCF8574配置 *****************************/ #define PCF8574_SLAVE_ADDR 0x23 // PCF8574 7位I2C地址(A0=1、A1=1、A2=0) #define PCF8574_IO_NUM 8 // 8个IO口(P0-P7) #define PCF8574_IO_DIR 0x0F // P0-P3输出(0),P4-P7输入(1) #define I2C_DEV_PATH "/dev/i2c-1"// 对应裸机的IIC1 /************************** 函数声明 *****************************/ int pcf8574_init(int fd); int pcf8574_set_output(int fd, uint8_t output_val); int pcf8574_get_input(int fd, uint8_t *input_val); int pcf8574_set_pin_level(int fd, uint8_t pin, uint8_t level); int i2c_write(int fd, uint8_t *buf, int len); int i2c_read(int fd, uint8_t *buf, int len); int main(int argc, char *argv[]) { int fd; uint8_t output_cnt = 0; uint8_t input_val, output_val; // 1. 打开I2C设备文件(/dev/i2c-1) fd = open(I2C_DEV_PATH, O_RDWR); if (fd < 0) { perror("Failed to open /dev/i2c-1"); return -1; } // 2. 设置PCF8574从机地址(0x23) if (ioctl(fd, I2C_SLAVE, PCF8574_SLAVE_ADDR) < 0) { perror("Failed to set PCF8574 slave address"); close(fd); return -1; } // 3. 初始化PCF8574(默认输出低电平) if (pcf8574_init(fd) != 0) { fprintf(stderr, "PCF8574 init failed\n"); close(fd); return -1; } printf("PCF8574 init success (Slave Addr: 0x%02X)\n", PCF8574_SLAVE_ADDR); // 4. 循环测试:对齐裸机逻辑(P0-P3输出循环,P4-P7读输入) while (1) { // 设置P0-P3输出电平(循环0-15) output_val = output_cnt % 16; if (pcf8574_set_output(fd, output_val) == 0) { printf("Set Output (P0-P3): 0x%02X (Binary: %04b)\n", output_val, output_val); } else { fprintf(stderr, "Set Output Failed\n"); } // 读取P4-P7输入电平 if (pcf8574_get_input(fd, &input_val) == 0) { // 屏蔽输出位,仅保留P4-P7(右移4位方便查看) input_val = (input_val & PCF8574_IO_DIR) >> 4; printf("Read Input (P4-P7): 0x%02X (Binary: %04b)\n", input_val, input_val); } else { fprintf(stderr, "Read Input Failed\n"); } // 示例:单独控制P1口电平(每2次循环翻转一次) if (output_cnt % 2 == 0) { pcf8574_set_pin_level(fd, 1, 1); // P1置高 printf("Set P1 Level: HIGH\n"); } else { pcf8574_set_pin_level(fd, 1, 0); // P1置低 printf("Set P1 Level: LOW\n"); } output_cnt++; sleep(1); // 1秒循环,对齐裸机 } close(fd); return 0; } /** * @brief 初始化PCF8574(默认输出低电平) * @param fd: I2C设备文件描述符 * @return 0成功,-1失败 */ int pcf8574_init(int fd) { return pcf8574_set_output(fd, 0x00); // 输出口默认低电平 } /** * @brief 设置PCF8574输出口电平(仅P0-P3有效) * @param fd: I2C设备文件描述符 * @param output_val: 输出值(仅低4位有效) * @return 0成功,-1失败 */ int pcf8574_set_output(int fd, uint8_t output_val) { uint8_t write_buf[1]; // 逻辑对齐裸机:输出口设为指定值,输入口设为1(高阻) write_buf[0] = (output_val & (~PCF8574_IO_DIR)) | PCF8574_IO_DIR; if (i2c_write(fd, write_buf, 1) != 1) { return -1; } usleep(1000); // 短暂等待总线稳定 return 0; } /** * @brief 读取PCF8574输入口电平(仅P4-P7有效) * @param fd: I2C设备文件描述符 * @param input_val: 存储输入值的指针 * @return 0成功,-1失败 */ int pcf8574_get_input(int fd, uint8_t *input_val) { uint8_t read_buf[1] = {0}; if (i2c_read(fd, read_buf, 1) != 1) { return -1; } *input_val = read_buf[0]; return 0; } /** * @brief 设置PCF8574指定输出Pin的电平(对齐裸机的SetPinLevel函数) * @param fd: I2C设备文件描述符 * @param pin: 目标Pin(0-7) * @param level: 电平(0=低,1=高) * @return 0成功,-1失败 */ int pcf8574_set_pin_level(int fd, uint8_t pin, uint8_t level) { // 检查Pin合法性+是否为输出口 if (pin >= PCF8574_IO_NUM || (PCF8574_IO_DIR & (1 << pin))) { fprintf(stderr, "Pin %d is invalid or not an output pin!\n", pin); return -1; } // 读取当前IO状态 uint8_t current_val; if (pcf8574_get_input(fd, &current_val) != 0) { return -1; } // 屏蔽输入口位,仅保留当前输出口状态 current_val &= ~PCF8574_IO_DIR; // 修改目标Pin电平 if (level == 0) { current_val &= ~(1 << pin); // 置低 } else { current_val |= (1 << pin); // 置高 } // 写回新状态 return pcf8574_set_output(fd, current_val); } /** * @brief Linux I2C写操作封装 * @param fd: I2C设备描述符 * @param buf: 写缓冲区 * @param len: 长度 * @return 实际写入长度,失败返回-1 */ int i2c_write(int fd, uint8_t *buf, int len) { int ret = write(fd, buf, len); if (ret < 0) { perror("I2C write failed"); return -1; } return ret; } /** * @brief Linux I2C读操作封装 * @param fd: I2C设备描述符 * @param buf: 读缓冲区 * @param len: 长度 * @return 实际读取长度,失败返回-1 */ int i2c_read(int fd, uint8_t *buf, int len) { int ret = read(fd, buf, len); if (ret < 0) { perror("I2C read failed"); return -1; } return ret; }
版权声明: 本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若内容造成侵权/违法违规/事实不符,请联系邮箱:809451989@qq.com进行投诉反馈,一经查实,立即删除!
网站建设 2026/5/10 21:33:46

通信系统仿真:数字调制与解调技术_(28).Simulink通信系统仿真

Simulink通信系统仿真 1. Simulink概述 1.1 Simulink简介 Simulink是MATLAB的一个附加产品&#xff0c;主要用于动态系统建模、仿真和分析。Simulink提供了一个图形用户界面&#xff08;GUI&#xff09;&#xff0c;使用户可以通过拖放操作来构建复杂的系统模型。在通信系统仿真…

作者头像 李华
网站建设 2026/5/10 21:33:54

论文AIGC率太高?5个超实用降AI工具分享,AI率降低80%以上

最近身边好多同学都在为论文的AI率头疼。明明自己认真改过好几遍了&#xff0c;查重系统还是显示AIGC率爆表&#xff0c;导师那关根本过不去。现在用AI工具辅助确实提升了效率&#xff0c;但怎么让内容更“人性化”&#xff0c;避免被系统标红&#xff0c;这才是关键。 我自己试…

作者头像 李华
网站建设 2026/5/10 21:33:28

团队合并与拆分的判断标准

团队合并与拆分的判断标准&#xff0c;核心是服务于组织的战略目标、效率和敏捷性。合并的判断标准主要基于“协同增效”和“规模效益”&#xff0c;当多个团队目标高度一致、资源重叠或需要集中力量办大事&#xff08;如攻坚核心项目&#xff09;时&#xff0c;应考虑合并。 而…

作者头像 李华
网站建设 2026/5/3 15:08:27

C#字典操作与类对象全解析

《字典》字典&#xff1a;包含了key&#xff08;键&#xff09;和value&#xff08;值&#xff09;&#xff0c;简称键值对&#xff0c;键是唯一的&#xff0c;值可以是任意类型的&#xff0c;字典是没有顺序的&#xff0c;只能通过键取值。 创建字典的方法 // Dictionary<…

作者头像 李华
网站建设 2026/5/9 17:34:44

智能脱机门禁考勤系统配置方案,核心设备为DAIC-KQ-ZW脱机考勤机,支持指纹/密码识别,可存储1000用户信息和5.5万条记录。系统具备U盘导出Excel报表功能,无需复杂软件,适合中小企业使用。

智能脱机门禁考勤系统配置清单及详细参数说明。&#x1f4cc; 设备型号概览设备类型型号主要功能脱机考勤机DAIC-KQ-ZW指纹/密码识别&#xff0c;U盘导出报表考勤管理系统DAIC-KQ3.0考勤软件管理门禁考勤服务器DAIC-MJ-SF门禁考勤服务支持指纹采集仪DAZW-iRead指纹信息采集人脸…

作者头像 李华