找回密码
 立即注册

QQ登录

只需一步,快速开始

搜索
查看: 8797|回复: 18
打印 上一主题 下一主题
收起左侧

关于单片机通过CH375芯片控制USB打印机的相关代码,有需要的可以参考一下哦

  [复制链接]
跳转到指定楼层
楼主

下载:
ch375print.zip (79.26 KB, 下载次数: 156)


部分代码预览:
  1. /*
  2. ****************************************
  3. **  Copyright  (C)  W.ch  1999-2005   **
  4. ****************************************
  5. **  USB 1.1 Host Examples for CH375   **
  6. **  KC7.0@MCS-51                      **
  7. ****************************************
  8. */
  9. /* 单片机通过CH375控制USB打印机 */
  10. /* 程序示例,C语言,CH375中断为查询方式,只负责数据传输,不涉及打印格式及打印描述语言 */
  11. /* 另可提供多台计算机共享一台USB打印机的方案 */

  12. /* 以下定义适用于MCS-51单片机,其它单片机参照修改,为了提供C语言的速度需要对本程序进行优化 */
  13. #include <reg51.h>
  14. unsigned char volatile xdata        CH375_CMD_PORT _at_ 0xFE00;        /* CH375命令端口的I/O地址 */
  15. unsigned char volatile xdata        CH375_DAT_PORT _at_ 0xFC00;        /* CH375数据端口的I/O地址 */
  16. sbit        CH375_INT_WIRE        =                0xB0^2;        /* P3.2, INT0, 连接CH375的INT#引脚,用于查询中断状态 */
  17. bit flag_config_2=0;
  18. bit flag_interface_2=0;                        //多个接口标志位
  19. typedef        unsigned char BOOL1;  /* typedef        bit        BOOL1; */


  20. /* 以下为通用的单片机C程序 */
  21. #include <string.h>
  22. #include <stdio.h>

  23. /* 定义CH375命令代码及返回状态 */
  24. #include "CH375INC.H"

  25. typedef unsigned char        UCHAR;
  26. typedef unsigned short        USHORT;
  27. union _REQUEST                                        //请求包结构
  28. {        struct
  29.         {        unsigned char bmRequestType;
  30.                 unsigned char bRequest;
  31.                 unsigned int wValue;
  32.                 unsigned int wIndex;
  33.                 unsigned int wLength;
  34.         }Req;
  35.         unsigned char Req_buf[8];
  36. }Request;


  37. typedef struct _USB_DEVICE_DESCRIPTOR {
  38.     UCHAR bLength;
  39.     UCHAR bDescriptorType;
  40.     USHORT bcdUSB;
  41.     UCHAR bDeviceClass;
  42.     UCHAR bDeviceSubClass;
  43.     UCHAR bDeviceProtocol;
  44.     UCHAR bMaxPacketSize0;
  45.     USHORT idVendor;
  46.     USHORT idProduct;
  47.     USHORT bcdDevice;
  48.     UCHAR iManufacturer;
  49.     UCHAR iProduct;
  50.     UCHAR iSerialNumber;
  51.     UCHAR bNumConfigurations;
  52. } USB_DEV_DESCR, *PUSB_DEV_DESCR;

  53. typedef struct _USB_CONFIG_DESCRIPTOR {
  54.     UCHAR bLength;
  55.     UCHAR bDescriptorType;
  56.     USHORT wTotalLength;
  57.     UCHAR bNumInterfaces;
  58.     UCHAR bConfigurationValue;
  59.     UCHAR iConfiguration;
  60.     UCHAR bmAttributes;
  61.     UCHAR MaxPower;
  62. } USB_CFG_DESCR, *PUSB_CFG_DESCR;

  63. typedef struct _USB_INTERF_DESCRIPTOR {
  64.     UCHAR bLength;
  65.     UCHAR bDescriptorType;
  66.     UCHAR bInterfaceNumber;
  67.     UCHAR bAlternateSetting;
  68.     UCHAR bNumEndpoints;
  69.     UCHAR bInterfaceClass;
  70.     UCHAR bInterfaceSubClass;
  71.     UCHAR bInterfaceProtocol;
  72.     UCHAR iInterface;
  73. } USB_ITF_DESCR, *PUSB_ITF_DESCR;

  74. typedef struct _USB_ENDPOINT_DESCRIPTOR {
  75.     UCHAR bLength;
  76.     UCHAR bDescriptorType;
  77.     UCHAR bEndpointAddress;
  78.     UCHAR bmAttributes;
  79.     UCHAR wMaxPacketSize;
  80.     UCHAR wMaxPacketSize1;
  81.     UCHAR bInterval;
  82. } USB_ENDP_DESCR, *PUSB_ENDP_DESCR;

  83. typedef struct _USB_CONFIG_DESCRIPTOR_LONG {
  84.         USB_CFG_DESCR        cfg_descr;
  85.         USB_ITF_DESCR        itf_descr;
  86. } USB_CFG_DESCR_LONG, *PUSB_CFG_DESCR_LONG;

  87. PUSB_ITF_DESCR  itf_descr;
  88. PUSB_ENDP_DESCR end_descr;

  89. unsigned char xdata buffer[256];                /* 公用缓冲区 */

  90. /* 延时2微秒,不精确 */
  91. void        delay2us( )
  92. {
  93.         unsigned char i;
  94.         for ( i = 2; i != 0; i -- );
  95. }

  96. /* 延时1微秒,不精确 */
  97. void        delay1us( )
  98. {
  99.         unsigned char i;
  100.         for ( i = 1; i != 0; i -- );
  101. }

  102. /* 以毫秒为单位延时,不精确,适用于24MHz时钟 */
  103. void        mDelaymS( unsigned char delay )
  104. {
  105.         unsigned char        i, j, c;
  106.         for ( i = delay; i != 0; i -- ) {
  107.                 for ( j = 200; j != 0; j -- ) c += 3;  /* 在24MHz时钟下延时500uS */
  108.                 for ( j = 200; j != 0; j -- ) c += 3;  /* 在24MHz时钟下延时500uS */
  109.         }
  110. }

  111. /* 基本操作 */

  112. void CH375_WR_CMD_PORT( unsigned char cmd ) {  /* 向CH375的命令端口写入命令,周期不小于4uS,如果单片机较快则延时 */
  113.         delay2us();
  114.         CH375_CMD_PORT=cmd;
  115.         delay2us();
  116. }

  117. void CH375_WR_DAT_PORT( unsigned char dat ) {  /* 向CH375的数据端口写入数据,周期不小于1.5uS,如果单片机较快则延时 */
  118.         CH375_DAT_PORT=dat;
  119.         delay1us();  /* 因为MCS51单片机较慢所以实际上无需延时 */
  120. }

  121. unsigned char CH375_RD_DAT_PORT() {  /* 从CH375的数据端口读出数据,周期不小于1.5uS,如果单片机较快则延时 */
  122.         delay1us();  /* 因为MCS51单片机较慢所以实际上无需延时 */
  123.         return( CH375_DAT_PORT );
  124. }

  125. unsigned char wait_interrupt() {  /* 主机端等待操作完成, 返回操作状态 */
  126.         unsigned short i;
  127. //        while( CH375_INT_WIRE );  /* 查询等待CH375操作完成中断(INT#低电平) */
  128.         for ( i = 0; CH375_INT_WIRE != 0; i ++ ) {  /* 如果CH375的中断引脚输出高电平则等待,通过计数防止超时 */
  129.                 delay1us();
  130.                 if ( i == 0xF000 ) CH375_WR_CMD_PORT( CMD_ABORT_NAK );  /* 如果超时达61mS以上则强行终止NAK重试,中断返回USB_INT_RET_NAK */
  131.         }

  132.         CH375_WR_CMD_PORT( CMD_GET_STATUS );  /* 产生操作完成中断, 获取中断状态 */
  133.         return( CH375_RD_DAT_PORT() );
  134. }

  135. #define        TRUE        1
  136. #define        FALSE        0
  137. unsigned char set_usb_mode( unsigned char mode ) {  /* 设置CH375的工作模式 */
  138.         unsigned char i;
  139.         CH375_WR_CMD_PORT( CMD_SET_USB_MODE );
  140.         CH375_WR_DAT_PORT( mode );
  141.         for( i=0; i!=100; i++ ) {  /* 等待设置模式操作完成,不超过30uS */
  142.                 if ( CH375_RD_DAT_PORT()==CMD_RET_SUCCESS ) return( TRUE );  /* 成功 */
  143.         }
  144.         return( FALSE );  /* CH375出错,例如芯片型号错或者处于串口方式或者不支持 */
  145. }

  146. /* 数据同步 */
  147. /* USB的数据同步通过切换DATA0和DATA1实现: 在设备端, USB打印机可以自动切换;
  148.    在主机端, 必须由SET_ENDP6和SET_ENDP7命令控制CH375切换DATA0与DATA1.
  149.    主机端的程序处理方法是为设备端的各个端点分别提供一个全局变量,
  150.    初始值均为DATA0, 每执行一次成功事务后取反, 每执行一次失败事务后将其复位为DATA1 */

  151. void toggle_recv( BOOL1 tog ) {  /* 主机接收同步控制:0=DATA0,1=DATA1 */
  152.         CH375_WR_CMD_PORT( CMD_SET_ENDP6 );
  153.         CH375_WR_DAT_PORT( tog ? 0xC0 : 0x80 );
  154.         delay2us();
  155. }

  156. void toggle_send( BOOL1 tog ) {  /* 主机发送同步控制:0=DATA0,1=DATA1 */
  157.         CH375_WR_CMD_PORT( CMD_SET_ENDP7 );
  158.         CH375_WR_DAT_PORT( tog ? 0xC0 : 0x80 );
  159.         delay2us();
  160. }

  161. unsigned char clr_stall( unsigned char endp_addr ) {  /* USB通讯失败后,复位设备端的指定端点到DATA0 */
  162.         CH375_WR_CMD_PORT( CMD_CLR_STALL );
  163.         CH375_WR_DAT_PORT( endp_addr );
  164.         return( wait_interrupt() );
  165. }

  166. /* 数据读写, 单片机读写CH375芯片中的数据缓冲区 */

  167. unsigned char rd_usb_data( unsigned char *buf ) {  /* 从CH37X读出数据块 */
  168.         unsigned char i, len;
  169.         CH375_WR_CMD_PORT( CMD_RD_USB_DATA );  /* 从CH375的端点缓冲区读取接收到的数据 */
  170.         len=CH375_RD_DAT_PORT();  /* 后续数据长度 */
  171.         for ( i=0; i!=len; i++ )
  172.          *buf++=CH375_RD_DAT_PORT();
  173.         return( len );
  174. }

  175. void wr_usb_data( unsigned char len, unsigned char *buf ) {  /* 向CH37X写入数据块 */
  176.         CH375_WR_CMD_PORT( CMD_WR_USB_DATA7 );  /* 向CH375的端点缓冲区写入准备发送的数据 */
  177.         CH375_WR_DAT_PORT( len );  /* 后续数据长度, len不能大于64 */
  178.         while( len-- ) CH375_WR_DAT_PORT( *buf++ );
  179. }

  180. /* 主机操作 */
  181. unsigned char endp_out_addr;        /* 打印机数据接收端点的端点地址 */
  182. unsigned char endp_out_size;        /* 打印机数据接收端点的端点尺寸 */
  183. BOOL1        tog_send;                                /* 打印机数据接收端点的同步标志 */
  184. unsigned char endp_in_addr;                /* 双向打印机发送端点的端点地址,一般不用 */
  185. BOOL1        tog_recv;                                /* 双向打印机发送端点的同步标志,一般不用 */

  186. unsigned char issue_token( unsigned char endp_and_pid ) {  /* 执行USB事务 */
  187. /* 执行完成后, 将产生中断通知单片机, 如果是USB_INT_SUCCESS就说明操作成功 */
  188.         CH375_WR_CMD_PORT( CMD_ISSUE_TOKEN );
  189.         CH375_WR_DAT_PORT( endp_and_pid );  /* 高4位目的端点号, 低4位令牌PID */
  190.         return( wait_interrupt() );  /* 等待CH375操作完成 */
  191.         //status=0xff;
  192. }


  193. void soft_reset_print( ) {  /* 控制传输:软复位打印机 */
  194.         tog_send=tog_recv=0;  /* 复位USB数据同步标志 */
  195.         toggle_send( 0 );  /* SETUP阶段为DATA0 */
  196.         buffer[0]=0x21; buffer[1]=2; buffer[2]=buffer[3]=0; buffer[4]=(itf_descr->bInterfaceNumber); buffer[5]=0; buffer[6]=buffer[7]=0;  /* SETUP数据,SOFT_RESET */
  197.         wr_usb_data( 8, buffer );  /* SETUP数据总是8字节 */
  198.         if ( issue_token( ( 0 << 4 ) | DEF_USB_PID_SETUP )==USB_INT_SUCCESS ) {  /* SETUP阶段操作成功 */
  199.                 toggle_recv( 1 );  /* STATUS阶段,准备接收DATA1 */
  200.         if ( issue_token( ( 0 << 4 ) | DEF_USB_PID_IN )==USB_INT_SUCCESS )
  201.                 return;  /* STATUS阶段操作成功,操作成功返回 */
  202.         }
  203. }

  204. #define        USB_INT_RET_NAK                0x2A                /* 00101010B,返回NAK */
  205. void send_data( unsigned short len, unsigned char *buf ) {  /* 主机发送数据块,一次最多64KB */
  206.         unsigned char l, s;
  207.         while( len ) {  /* 连续输出数据块给USB打印机 */
  208.                 toggle_send( tog_send );  /* 数据同步 */
  209.                 l = len>endp_out_size?endp_out_size:len;  /* 单次发送不能超过端点尺寸 */
  210.                 wr_usb_data( l, buf );  /* 将数据先复制到CH375芯片中 */
  211.                 s = issue_token( ( endp_out_addr << 4 ) | DEF_USB_PID_OUT );  /* 请求CH375输出数据 */
  212.                 if ( s==USB_INT_SUCCESS ) {  /* CH375成功发出数据 */
  213.                         tog_send = ~ tog_send;  /* 切换DATA0和DATA1进行数据同步 */
  214.                         len-=l;  /* 计数 */
  215.                         buf+=l;  /* 操作成功 */
  216.                 }
  217.                 else if ( s==USB_INT_RET_NAK ) {  /* USB打印机正忙,如果未执行SET_RETRY命令则CH375自动重试,所以不会返回USB_INT_RET_NAK状态 */
  218.                         /* USB打印机正忙,正常情况下应该稍后重试 */
  219.                         /* s=get_port_status( );  如果有必要,可以检查是什么原因导致打印机忙 */
  220.                 }
  221.                 else {  /* 操作失败,正常情况下不会失败 */
  222.                         clr_stall( endp_out_addr );  /* 清除打印机的数据接收端点,或者 soft_reset_print() */
  223. /*                        soft_reset_print();  打印机出现意外错误,软复位 */
  224.                         tog_send = 0;  /* 操作失败 */
  225.                 }
  226. /* 如果数据量较大,可以定期调用get_port_status()检查打印机状态 */
  227.         }
  228. }

  229. unsigned char get_port_status( ) {  /* 查询打印机端口状态,返回状态码,如果为0FFH则说明操作失败 */
  230. /* 返回状态码中: 位5(Paper Empty)为1说明无纸, 位4(Select)为1说明打印机联机, 位3(Not Error)为0说明打印机出错 */
  231.         toggle_send( 0 );  /* 下面通过控制传输获取打印机的状态, SETUP阶段为DATA0 */
  232.         buffer[0]=0xA1; buffer[1]=1; buffer[2]=buffer[3]=0; buffer[4]=(itf_descr->bInterfaceNumber); buffer[5]=0; buffer[6]=1; buffer[7]=0;  /* SETUP数据,GET_PORT_STATUS */
  233.         wr_usb_data( 8, buffer );  /* SETUP数据总是8字节 */
  234.         if ( issue_token( ( 0 << 4 ) | DEF_USB_PID_SETUP )==USB_INT_SUCCESS ) {  /* SETUP阶段操作成功 */
  235.                 toggle_recv( 1 );  /* DATA阶段,准备接收DATA1 */
  236.                 if ( issue_token( ( 0 << 4 ) | DEF_USB_PID_IN )==USB_INT_SUCCESS ) {  /* DATA阶段操作成功 */
  237.                         rd_usb_data( buffer );  /* 读出接收到的数据,通常只有1个字节 */
  238.                         toggle_send( 1 );  /* STATUS阶段为DATA1 */
  239.                         wr_usb_data( 0, buffer );  /* 发送0长度的数据说明控制传输成功 */
  240.                         if ( issue_token( ( 0 << 4 ) | DEF_USB_PID_OUT )==USB_INT_SUCCESS )
  241.                                  return( buffer[0] );  /* 返回状态码 */
  242.                 }
  243.         }
  244.         return( 0xFF );  /* 返回操作失败 */
  245. }


  246. unsigned char get_descr( unsigned char type ) {  /* 从设备端获取描述符 */
  247.         CH375_WR_CMD_PORT( CMD_GET_DESCR );
  248.         CH375_WR_DAT_PORT( type );  /* 描述符类型, 只支持1(设备)或者2(配置) */
  249.         return( wait_interrupt() );  /* 等待CH375操作完成 */
  250. }


  251. unsigned char set_addr( unsigned char addr ) {  /* 设置设备端的USB地址 */
  252.         unsigned char status;
  253.         CH375_WR_CMD_PORT( CMD_SET_ADDRESS );  /* 设置USB设备端的USB地址 */
  254.         CH375_WR_DAT_PORT( addr );  /* 地址, 从1到127之间的任意值, 常用2到20 */
  255.         status=wait_interrupt();  /* 等待CH375操作完成 */
  256.         if ( status==USB_INT_SUCCESS ) {  /* 操作成功 */
  257.                 CH375_WR_CMD_PORT( CMD_SET_USB_ADDR );  /* 设置USB主机端的USB地址 */
  258.                 CH375_WR_DAT_PORT( addr );  /* 当目标USB设备的地址成功修改后,应该同步修改主机端的USB地址 */
  259.         }
  260.         mDelaymS( 5 );
  261.         return( status );
  262. }

  263. …………限于本文篇幅 余下代码请从51黑下载附件…………
复制代码



分享到:  QQ好友和群QQ好友和群 QQ空间QQ空间 腾讯微博腾讯微博 腾讯朋友腾讯朋友
收藏收藏3 分享淘帖 顶 踩
回复

使用道具 举报

来自 2#
ID:238057 发表于 2017-10-9 17:33 | 只看该作者
请问你驱动的是什么打印机,我驱动热敏打印机用你的代码测试,调试发现get_descr_ex()函数返回为0
回复

使用道具 举报

板凳
ID:59768 发表于 2017-5-29 17:52 | 只看该作者
有空下来看看
回复

使用道具 举报

地板
ID:212056 发表于 2017-6-17 13:21 | 只看该作者
感觉代码质量很好啊 正好有需要! 感谢
回复

使用道具 举报

5#
ID:222124 发表于 2017-7-24 20:22 | 只看该作者
有项目用到这个,学习一下,谢谢!
回复

使用道具 举报

6#
ID:233727 发表于 2017-10-20 10:04 | 只看该作者
顶一下,
回复

使用道具 举报

7#
ID:233727 发表于 2017-11-5 14:06 | 只看该作者
        unsigned char str_to_print[]= {27,'E'};
        unsigned char wdf_pcl[] = {27,'E','a', 'b', 27,'&', 'l', '-',55, 48, 'C' };

请问大侠,这些数据的含义是?搞不懂。谢谢大侠。
回复

使用道具 举报

8#
ID:280640 发表于 2018-1-29 16:14 | 只看该作者
零度的亲吻 发表于 2017-10-9 17:33
请问你驱动的是什么打印机,我驱动热敏打印机用你的代码测试,调试发现get_descr_ex()函数返回为0

估计不同打印机要求的数据格式不一样,这里作者在程序开头就提到了
/* 程序示例,C语言,CH375中断为查询方式,只负责数据传输,不涉及打印格式及打印描述语言 */
具体应该需要去查询你所使用的打印机的通信协议。
回复

使用道具 举报

9#
ID:280640 发表于 2018-1-29 16:14 | 只看该作者
很棒的例子,正好需要用到,学习了学习了
回复

使用道具 举报

10#
ID:282409 发表于 2018-2-5 17:14 | 只看该作者
大家好,CH375作为USB HOST可以驱动USB打印机,但也只是为USB主机和USB打印机搭建USB通信通道,但是至于USB打印机的打印协议还是跟各个打印机厂家相关,不同厂家的打印协议通常会不太一样(包括打印初始化等命令)所以不可能做到一段代码能兼容到所有USB打印机,这也是嵌入式系统操作USB打印机最大的难点,针对不同的打印机还是需要针对性的修改代码。
回复

使用道具 举报

11#
ID:290882 发表于 2018-6-10 16:30 | 只看该作者
如果文件大于64k怎么办。
回复

使用道具 举报

12#
ID:388192 发表于 2018-8-23 20:16 | 只看该作者
非常好的资料,感谢
回复

使用道具 举报

13#
ID:473553 发表于 2019-4-25 09:04 | 只看该作者
借鉴参考一下,非常感谢
回复

使用道具 举报

14#
ID:473553 发表于 2019-4-25 09:34 | 只看该作者
哪位老师能给解答一下        CH375_WR_CMD_PORT( CMD_WR_USB_DATA7 );  /* 向CH375的端点缓冲区写入准备发送的数据 */
                                                CH375_WR_DAT_PORT(5);
                                                CH375_WR_DAT_PORT(9);
                                                CH375_WR_DAT_PORT(3);
                                                CH375_WR_DAT_PORT(5);
                                                CH375_WR_DAT_PORT(6);
                                                        CH375_WR_DAT_PORT(7);

应该是发送5个字节的数据,我把发发送到usb转串口上。,用逻辑分析仪拦截,结果只发送出去了3个字节,剩下两个丢掉了,为什么?
回复

使用道具 举报

15#
ID:494850 发表于 2019-6-9 10:18 | 只看该作者
收了,学习下
回复

使用道具 举报

16#
ID:244210 发表于 2020-6-6 20:02 | 只看该作者
下载看看,谢谢!
回复

使用道具 举报

17#
ID:458012 发表于 2020-11-11 13:55 | 只看该作者
MARK~~~~
回复

使用道具 举报

18#
ID:458012 发表于 2020-11-11 13:56 | 只看该作者
收藏一下,后续用到的时候来看看
回复

使用道具 举报

19#
ID:30192 发表于 2023-10-8 16:33 | 只看该作者
下载看看,万一以后会用到
回复

使用道具 举报

您需要登录后才可以回帖 登录 | 立即注册

本版积分规则

手机版|小黑屋|51黑电子论坛 |51黑电子论坛6群 QQ 管理员QQ:125739409;技术交流QQ群281945664

Powered by 单片机教程网

快速回复 返回顶部 返回列表