用于规范模式状态的Termios手册页(http://man7.org/linux/man-
pages/man3/termios.3.html
):
逐行提供输入。输入一个行定界符(NL,EOL,EOL2;或在行首的EOF)时,输入行可用。除EOF以外,行定界符包含在read(2)返回的缓冲区中。
我的问题是:当一块硬件输出符合规范的数据时,是否将0xD0xA(CRLF)字节放在传输线的开头,以告知read()函数可以读取数据了?
我之前并没有对此进行过多考虑,并且已经默认(可能错误地)认为0xD0xA位于传输线的末端。
是否将0xD0xA(CRLF)字节放在传输线的开头以告知read()函数数据已准备好被读取?
在 “串行端口” 或 “硬件” 没有一个概念 “开始” 或 “结束” 的的 “传输线” 。只是到U [S] ART的有效载荷数据。
仅当在标准模式下使用termios读取串行终端缓冲区时,线路终端才具有上下文。
请参阅Linux串行驱动程序,以了解如何从硬件中删除用户空间代码。
Linux使用换行符或具有ASCII代码0x0A的换行符作为行终止符,如 手册 页(已引述)中明确指出的那样。
Termios允许定义其他行尾字符,即串行终端的VEOL和VEOL2。
每次出现行定界符都会并且将导致(待定)规范 read() 返回。
行定界符将是缓冲区中返回的最后一个字符,除非用户缓冲区太小而无法包含整行。
为EOF定义的字符,即 VEOF ,默认为EOT的ASCII代码0x04,由termios处理略有不同。
EOF字符的接收导致(待定的)规范 read() 像行定界符一样返回,但是EOF字符未存储在返回的缓冲区中。
因此,当EOF前面有行定界符时, read() 的返回码为零,即实际的空行!
如果您是一位怀疑的托马斯,那么您应该将一对USB-RS232适配器交叉连接在一起,并测试使用termios从串行终端读取数据时会发生什么。在第一个串行终端上
使用终端仿真器程序(例如 minicom) 输入数据,并使用以下C程序查看另一个串行终端上的规范读取。
#define SERIALTERMINAL "/dev/ttyUSB1"
#include <errno.h>
#include <fcntl.h>
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <termios.h>
#include <unistd.h>
int set_interface_attribs(int fd, int speed)
{
struct termios tty;
if (tcgetattr(fd, &tty) < 0) {
printf("Error from tcgetattr: %s\n", strerror(errno));
return -1;
}
cfsetospeed(&tty, (speed_t)speed);
cfsetispeed(&tty, (speed_t)speed);
tty.c_cflag |= CLOCAL | CREAD;
tty.c_cflag &= ~CSIZE;
tty.c_cflag |= CS8; /* 8-bit characters */
tty.c_cflag &= ~PARENB; /* no parity bit */
tty.c_cflag &= ~CSTOPB; /* only need 1 stop bit */
tty.c_cflag &= ~CRTSCTS; /* no hardware flowcontrol */
tty.c_lflag |= ICANON | ISIG; /* canonical input */
tty.c_lflag &= ~(ECHO | ECHOE | ECHONL | IEXTEN);
tty.c_iflag &= ~IGNCR; /* preserve carriage return */
tty.c_iflag &= ~INPCK;
tty.c_iflag &= ~(INLCR | ICRNL | IUCLC | IMAXBEL);
tty.c_iflag &= ~(IXON | IXOFF | IXANY); /* no SW flowcontrol */
tty.c_oflag &= ~OPOST;
tty.c_cc[VEOL] = 0;
tty.c_cc[VEOL2] = 0;
tty.c_cc[VEOF] = 0x04;
if (tcsetattr(fd, TCSANOW, &tty) != 0) {
printf("Error from tcsetattr: %s\n", strerror(errno));
return -1;
}
return 0;
}
int main()
{
char *portname = SERIALTERMINAL;
int fd;
int wlen;
fd = open(portname, O_RDWR | O_NOCTTY | O_SYNC);
if (fd < 0) {
printf("Error opening %s: %s\n", portname, strerror(errno));
return -1;
}
/*baudrate 115200, 8 bits, no parity, 1 stop bit */
set_interface_attribs(fd, B115200);
/* simple output */
wlen = write(fd, "Hello!\n", 7);
if (wlen != 7) {
printf("Error from write: %d, %d\n", wlen, errno);
}
tcdrain(fd); /* delay for output */
/* simple canonical input */
do {
unsigned char buf[81];
unsigned char *p;
int rdlen;
rdlen = read(fd, buf, sizeof(buf) - 1);
if (rdlen > 0) {
buf[rdlen] = 0;
printf("Read %d:", rdlen);
/* first display as hex numbers then ASCII */
for (p = buf; rdlen-- > 0; p++) {
printf(" 0x%x", *p);
if (*p < ' ')
*p = '.'; /* replace any control chars */
}
printf("\n \"%s\"\n\n", buf);
} else if (rdlen < 0) {
printf("Error from read: %d: %s\n", rdlen, strerror(errno));
} else { /* rdlen == 0 */
printf("Nothing read. EOF?\n");
}
/* repeat read */
} while (1);
}
请注意,该程序不会去除’\ r’字符(即,属性IGNCR被清除),但是回车符也未定义为行定界符。
因此,在此termios配置中,回车符没有特殊含义,并且像任何可打印字符一样通过。
因此,键入(等效于)ABCDEFG^M^J
的内容为:
Read 9: 0x41 0x42 0x43 0x44 0x45 0x46 0x47 0xd 0xa
"ABCDEFG.."
123^Mabc^J
读取为:
Read 8: 0x31 0x32 0x33 0xd 0x61 0x62 0x63 0xa
"123.abc."
可选的termios配置可以去除回车符或将回车符视为行定界符。
在一个应用软件的成型过程中,一些意想不到的商业逻辑到处出现。比如,基于价格的考虑,这个任务必须减少项目;而那个任务也因为销售税而必须选择合适的比率;而其它的任务也必须因为其他的特别条件而终止。一些商业规则是简单的,只需要不到一两个布尔比较关系就够了,然而它的规则可能需要费时的估计,需要查询数据库或者用户输入数据来引导。 通过书写代码可以把抽象(比如一条商业规则)转化为具体可见的东西。但是抽象物(比
公司及产品名称 我们的公司及产品名称是「DaoCloud」。注意这是两个单词的合成词,所以中间沒有空格(参考:GitHub)。如作为 URL 的一部分,应该使用全小写的「daocloud」。 文案风格 一定多检查,确保没有错别字。 即使是流行语中的谐音错别字也不要使用,比如「墙裂」、「童鞋」等。 我们崇尚精练的文风。请在检查中把对表达意思没有明显作用的字、词、句删除,在不影响表达效果的前提下把文案
前端规范 目的 旨在增强团队开发协作、提高代码质量和打造开发基石的编码规范,以下规范是团队基本约定的内容,必须严格遵循。 HTML 规范 基于 W3C 等官方文档,并结合团队业务和开发过程中总结的规范约定,让页面 HTML 代码更具语义性。 图片规范 了解各种图片格式特性,根据特性制定图片规范,包括但不限于图片的质量约定、图片引入方式、图片合并处理等,旨在从图片层面优化页面性能。 CSS 规范、命
coolie 遵循的是 CMD 规范。 CMD 规范 CMD 最先是由 seajs 提出的,是 commonJS 规范的前端实现。 与 commonJS 的表现是一致的,即:依赖就近,顺序执行。 define(function(require, exports, module){ var xhr = require('./xhr.js'); xhr.ajax(...); });
页面模块化 规范 在具体谈论规范的之前,可以下去查看下各大网络公司的前端开发规范(Developemnt Style Guide)例如谷歌,Facebook 或者 Dropbox。从而更好的理解开发规范在实际应用中和多人协作中的重要性。 不同开发者在开发过程中使用不同的代码风格会直接的提升在之后的开发和维护的成本和难度,对前端开发来说更是尤为突出。这时使用代码规范来约束开发者的编码风格就可以大体解
代码规范 代码风格 以 cargo 输出没有 warning 为准 可以通过根目录的 make fmt 来自动调用 cargo fmt 规范全部的代码 注释规范 用 //! 注释外层内容,例如在文件开始注释整个模块 用 /// 为函数添加 doc 注释,其内部使用 Markdown 语法 可以使用 markdown 格式的链接,链接内容使用 Rust 可以直接链上,例如 /// 样例注释 ///