Always strip/add parity if it is detected.

This commit is contained in:
Chris Osborn 2016-05-25 09:55:08 -07:00
parent 4b3dcc20d0
commit 04e0abc7f7
4 changed files with 42 additions and 36 deletions

View File

@ -1,4 +1,5 @@
This is yet another fork of Jim Brain's tcpser serial to IP modem emulation program. This is yet another fork of Jim Brain's tcpser serial to IP modem emulation program.
The original source code can be found here: The original source code can be found here:
http://www.jbrain.com/pub/linux/serial/ http://www.jbrain.com/pub/linux/serial/
@ -6,12 +7,12 @@ My changes are based upon the rc12 archive dated 11Mar09.
I fixed the bug with being unable to connect to real telnet servers. I fixed the bug with being unable to connect to real telnet servers.
I also made the modem routines automatically detect parity and ignore I also made the modem routines automatically detect parity. If even,
it in AT commands and print out modem responses in matching odd, or mark parity is detected then the telnet connection will be
parity. Parity is *not* stripped when sending data over the limited to 7 bit. Space parity will be treated as 8N1 and will allow
connection, which is how a real modem behaves. This may or may not be telnet binary mode.
what you want. Some servers will expect an 8 bit connection and may
not work. I also incorporated geneb's changes found at https://github.com/geneb/tcpser
Chris Osborn <fozztexx@fozztexx.com> Chris Osborn <fozztexx@fozztexx.com>
http://insentricity.com http://insentricity.com

View File

@ -385,7 +385,6 @@ void *run_bridge(void *arg)
} }
else else
mdm_send_ring(cfg); mdm_send_ring(cfg);
} }
else else
mdm_handle_timeout(cfg); mdm_handle_timeout(cfg);

View File

@ -22,9 +22,13 @@ int line_write(modem_config *cfg, char *data, int len)
int double_iac = FALSE; int double_iac = FALSE;
char text[1024]; char text[1024];
int text_len = 0; int text_len = 0;
int mask = 0x7f;
if (cfg->line_data.is_telnet && cfg->line_data.nvt_data.binary_xmit) { if (cfg->line_data.is_telnet) {
if (cfg->line_data.nvt_data.binary_xmit)
mask = 0xff;
retval = 0; retval = 0;
while (i < len) { while (i < len) {
if (double_iac) { if (double_iac) {
@ -38,10 +42,10 @@ int line_write(modem_config *cfg, char *data, int len)
double_iac = TRUE; double_iac = TRUE;
} }
else { else {
text[text_len++] = data[i++]; text[text_len++] = data[i++] & mask;
} }
} }
if (text_len == 1024) { if (text_len == sizeof(text_len)) {
retval = ip_write(cfg->line_data.fd, text, text_len); retval = ip_write(cfg->line_data.fd, text, text_len);
text_len = 0; text_len = 0;
} }

View File

@ -178,14 +178,7 @@ void mdm_write_char(modem_config *cfg, char data)
mdm_write(cfg, str, 1); mdm_write(cfg, str, 1);
} }
void mdm_write(modem_config *cfg, char data[], int len) void mdm_write(modem_config *cfg, char *data, int len)
{
if (cfg->allow_transmit == TRUE) {
dce_write(cfg, data, len);
}
}
void mdm_write_parity(modem_config *cfg, char *data, int len)
{ {
unsigned char *buf; unsigned char *buf;
int i; int i;
@ -193,18 +186,23 @@ void mdm_write_parity(modem_config *cfg, char *data, int len)
if (cfg->allow_transmit == TRUE) { if (cfg->allow_transmit == TRUE) {
if (cfg->parity) {
buf = malloc(len); buf = malloc(len);
memcpy(buf, data, len); memcpy(buf, data, len);
for (i = 0; i < len; i++) { for (i = 0; i < len; i++) {
v = buf[i]; v = buf[i];
v = (((v * 0x0101010101010101ULL) & 0x8040201008040201ULL) % 0x1FF) & 1; v = (((v * 0x0101010101010101ULL) & 0x8040201008040201ULL) % 0x1FF) & 1;
buf[i] &= 0x7f;
buf[i] |= (((cfg->parity >> !v)) & 1) << 7; buf[i] |= (((cfg->parity >> !v)) & 1) << 7;
} }
mdm_write(cfg, (char *) buf, len); dce_write(cfg, (char *) buf, len);
free(buf); free(buf);
} }
else
dce_write(cfg, data, len);
}
} }
void mdm_send_response(int msg, modem_config *cfg) void mdm_send_response(int msg, modem_config *cfg)
@ -214,17 +212,17 @@ void mdm_send_response(int msg, modem_config *cfg)
LOG(LOG_DEBUG, "Sending %s response to modem", mdm_responses[msg]); LOG(LOG_DEBUG, "Sending %s response to modem", mdm_responses[msg]);
if (cfg->send_responses == TRUE) { if (cfg->send_responses == TRUE) {
mdm_write_parity(cfg, cfg->crlf, 2); mdm_write(cfg, cfg->crlf, 2);
if (cfg->text_responses == TRUE) { if (cfg->text_responses == TRUE) {
LOG(LOG_ALL, "Sending text response"); LOG(LOG_ALL, "Sending text response");
mdm_write_parity(cfg, mdm_responses[msg], strlen(mdm_responses[msg])); mdm_write(cfg, mdm_responses[msg], strlen(mdm_responses[msg]));
} }
else { else {
LOG(LOG_ALL, "Sending numeric response"); LOG(LOG_ALL, "Sending numeric response");
sprintf(msgID, "%d", msg); sprintf(msgID, "%d", msg);
mdm_write_parity(cfg, msgID, strlen(msgID)); mdm_write(cfg, msgID, strlen(msgID));
} }
mdm_write_parity(cfg, cfg->crlf, 2); mdm_write(cfg, cfg->crlf, 2);
} }
} }
@ -389,8 +387,8 @@ int mdm_parse_cmd(modem_config *cfg)
strncpy(cfg->dialno, cfg->last_dialno, strlen(cfg->last_dialno)); strncpy(cfg->dialno, cfg->last_dialno, strlen(cfg->last_dialno));
cfg->dial_type = cfg->dial_type; cfg->dial_type = cfg->dial_type;
cfg->memory_dial = TRUE; cfg->memory_dial = TRUE;
mdm_write_parity(cfg, cfg->crlf, 2); mdm_write(cfg, cfg->crlf, 2);
mdm_write_parity(cfg, cfg->dialno, strlen(cfg->dialno)); mdm_write(cfg, cfg->dialno, strlen(cfg->dialno));
} }
else { else {
cfg->dialno[0] = 0; cfg->dialno[0] = 0;
@ -477,7 +475,7 @@ int mdm_parse_cmd(modem_config *cfg)
break; break;
case AT_CMD_FLAG_QUERY | 'S': case AT_CMD_FLAG_QUERY | 'S':
sprintf(tmp, "%s%3.3d", cfg->crlf, cfg->s[num]); sprintf(tmp, "%s%3.3d", cfg->crlf, cfg->s[num]);
mdm_write_parity(cfg, tmp, strlen(tmp)); mdm_write(cfg, tmp, strlen(tmp));
break; break;
case 'T': // defaut to tone dialing case 'T': // defaut to tone dialing
//cfg->default_dial_type=MDM_DT_TONE; //cfg->default_dial_type=MDM_DT_TONE;
@ -576,7 +574,7 @@ int mdm_handle_char(modem_config *cfg, char ch)
if (cfg->cmd_started == TRUE) { if (cfg->cmd_started == TRUE) {
if (ch == (char) (cfg->s[5])) { if (ch == (char) (cfg->s[5])) {
if (cfg->cur_line_idx == 0 && cfg->echo == TRUE) { if (cfg->cur_line_idx == 0 && cfg->echo == TRUE) {
mdm_write_parity(cfg, "T", 1); mdm_write(cfg, "T", 1);
} }
else { else {
cfg->cur_line_idx--; cfg->cur_line_idx--;
@ -631,6 +629,7 @@ int mdm_clear_break(modem_config *cfg)
int mdm_parse_data(modem_config *cfg, char *data, int len) int mdm_parse_data(modem_config *cfg, char *data, int len)
{ {
int i; int i;
int ch;
if (cfg->cmd_mode == TRUE) { if (cfg->cmd_mode == TRUE) {
@ -642,7 +641,10 @@ int mdm_parse_data(modem_config *cfg, char *data, int len)
line_write(cfg, data, len); line_write(cfg, data, len);
if (cfg->pre_break_delay == TRUE) { if (cfg->pre_break_delay == TRUE) {
for (i = 0; i < len; i++) { for (i = 0; i < len; i++) {
if (data[i] == (char) cfg->s[2]) { ch = data[i];
if (cfg->parity)
ch &= 0x7f;
if (ch == (char) cfg->s[2]) {
LOG(LOG_DEBUG, "Break character received"); LOG(LOG_DEBUG, "Break character received");
cfg->break_len++; cfg->break_len++;
if (cfg->break_len > 3) { // more than 3, considered invalid if (cfg->break_len > 3) { // more than 3, considered invalid