chardev: Basic support for TN3270
This introduces basic support for TN3270, which needs to negotiate three Telnet options during handshake: - End of Record - Binary Transmission - Terminal-Type As a basic implementation, this simply ignores NOP and Interrupt Process(IP) commands. More work should be done for them later. For more details, please refer to RFC 854 and 1576. Signed-off-by: Jing Liu <liujbjl@linux.vnet.ibm.com> Signed-off-by: Yang Chen <bjcyang@linux.vnet.ibm.com> Reviewed-by: QingFeng Hao <haoqf@linux.vnet.ibm.com> Acked-by: Dong Jia Shi <bjsdjshi@linux.vnet.ibm.com> Signed-off-by: Cornelia Huck <cornelia.huck@de.ibm.com> Reviewed-by: Marc-André Lureau <marcandre.lureau@redhat.com>
This commit is contained in:
parent
e619b14746
commit
ae92cbd542
@ -55,6 +55,7 @@ typedef struct {
|
|||||||
SocketAddress *addr;
|
SocketAddress *addr;
|
||||||
bool is_listen;
|
bool is_listen;
|
||||||
bool is_telnet;
|
bool is_telnet;
|
||||||
|
bool is_tn3270;
|
||||||
|
|
||||||
guint reconnect_timer;
|
guint reconnect_timer;
|
||||||
int64_t reconnect_time;
|
int64_t reconnect_time;
|
||||||
@ -141,19 +142,25 @@ static int tcp_chr_read_poll(void *opaque)
|
|||||||
return s->max_size;
|
return s->max_size;
|
||||||
}
|
}
|
||||||
|
|
||||||
#define IAC 255
|
|
||||||
#define IAC_BREAK 243
|
|
||||||
static void tcp_chr_process_IAC_bytes(Chardev *chr,
|
static void tcp_chr_process_IAC_bytes(Chardev *chr,
|
||||||
SocketChardev *s,
|
SocketChardev *s,
|
||||||
uint8_t *buf, int *size)
|
uint8_t *buf, int *size)
|
||||||
{
|
{
|
||||||
/* Handle any telnet client's basic IAC options to satisfy char by
|
/* Handle any telnet or tn3270 client's basic IAC options.
|
||||||
* char mode with no echo. All IAC options will be removed from
|
* For telnet options, it satisfies char by char mode with no echo.
|
||||||
* the buf and the do_telnetopt variable will be used to track the
|
* For tn3270 options, it satisfies binary mode with EOR.
|
||||||
* state of the width of the IAC information.
|
* All IAC options will be removed from the buf and the do_opt
|
||||||
|
* pointer will be used to track the state of the width of the
|
||||||
|
* IAC information.
|
||||||
*
|
*
|
||||||
* IAC commands come in sets of 3 bytes with the exception of the
|
* RFC854: "All TELNET commands consist of at least a two byte sequence.
|
||||||
* "IAC BREAK" command and the double IAC.
|
* The commands dealing with option negotiation are three byte sequences,
|
||||||
|
* the third byte being the code for the option referenced."
|
||||||
|
* "IAC BREAK", "IAC IP", "IAC NOP" and the double IAC are two bytes.
|
||||||
|
* "IAC SB", "IAC SE" and "IAC EOR" are saved to split up data boundary
|
||||||
|
* for tn3270.
|
||||||
|
* NOP, Break and Interrupt Process(IP) might be encountered during a TN3270
|
||||||
|
* session, and NOP and IP need to be done later.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
int i;
|
int i;
|
||||||
@ -174,6 +181,18 @@ static void tcp_chr_process_IAC_bytes(Chardev *chr,
|
|||||||
/* Handle IAC break commands by sending a serial break */
|
/* Handle IAC break commands by sending a serial break */
|
||||||
qemu_chr_be_event(chr, CHR_EVENT_BREAK);
|
qemu_chr_be_event(chr, CHR_EVENT_BREAK);
|
||||||
s->do_telnetopt++;
|
s->do_telnetopt++;
|
||||||
|
} else if (s->is_tn3270 && ((unsigned char)buf[i] == IAC_EOR
|
||||||
|
|| (unsigned char)buf[i] == IAC_SB
|
||||||
|
|| (unsigned char)buf[i] == IAC_SE)
|
||||||
|
&& s->do_telnetopt == 2) {
|
||||||
|
buf[j++] = IAC;
|
||||||
|
buf[j++] = buf[i];
|
||||||
|
s->do_telnetopt++;
|
||||||
|
} else if (s->is_tn3270 && ((unsigned char)buf[i] == IAC_IP
|
||||||
|
|| (unsigned char)buf[i] == IAC_NOP)
|
||||||
|
&& s->do_telnetopt == 2) {
|
||||||
|
/* TODO: IP and NOP need to be implemented later. */
|
||||||
|
s->do_telnetopt++;
|
||||||
}
|
}
|
||||||
s->do_telnetopt++;
|
s->do_telnetopt++;
|
||||||
}
|
}
|
||||||
@ -512,7 +531,7 @@ static void tcp_chr_update_read_handler(Chardev *chr,
|
|||||||
|
|
||||||
typedef struct {
|
typedef struct {
|
||||||
Chardev *chr;
|
Chardev *chr;
|
||||||
char buf[12];
|
char buf[21];
|
||||||
size_t buflen;
|
size_t buflen;
|
||||||
} TCPChardevTelnetInit;
|
} TCPChardevTelnetInit;
|
||||||
|
|
||||||
@ -550,9 +569,6 @@ static void tcp_chr_telnet_init(Chardev *chr)
|
|||||||
TCPChardevTelnetInit *init = g_new0(TCPChardevTelnetInit, 1);
|
TCPChardevTelnetInit *init = g_new0(TCPChardevTelnetInit, 1);
|
||||||
size_t n = 0;
|
size_t n = 0;
|
||||||
|
|
||||||
init->chr = chr;
|
|
||||||
init->buflen = 12;
|
|
||||||
|
|
||||||
#define IACSET(x, a, b, c) \
|
#define IACSET(x, a, b, c) \
|
||||||
do { \
|
do { \
|
||||||
x[n++] = a; \
|
x[n++] = a; \
|
||||||
@ -560,12 +576,26 @@ static void tcp_chr_telnet_init(Chardev *chr)
|
|||||||
x[n++] = c; \
|
x[n++] = c; \
|
||||||
} while (0)
|
} while (0)
|
||||||
|
|
||||||
/* Prep the telnet negotion to put telnet in binary,
|
init->chr = chr;
|
||||||
* no echo, single char mode */
|
if (!s->is_tn3270) {
|
||||||
IACSET(init->buf, 0xff, 0xfb, 0x01); /* IAC WILL ECHO */
|
init->buflen = 12;
|
||||||
IACSET(init->buf, 0xff, 0xfb, 0x03); /* IAC WILL Suppress go ahead */
|
/* Prep the telnet negotion to put telnet in binary,
|
||||||
IACSET(init->buf, 0xff, 0xfb, 0x00); /* IAC WILL Binary */
|
* no echo, single char mode */
|
||||||
IACSET(init->buf, 0xff, 0xfd, 0x00); /* IAC DO Binary */
|
IACSET(init->buf, 0xff, 0xfb, 0x01); /* IAC WILL ECHO */
|
||||||
|
IACSET(init->buf, 0xff, 0xfb, 0x03); /* IAC WILL Suppress go ahead */
|
||||||
|
IACSET(init->buf, 0xff, 0xfb, 0x00); /* IAC WILL Binary */
|
||||||
|
IACSET(init->buf, 0xff, 0xfd, 0x00); /* IAC DO Binary */
|
||||||
|
} else {
|
||||||
|
init->buflen = 21;
|
||||||
|
/* Prep the TN3270 negotion based on RFC1576 */
|
||||||
|
IACSET(init->buf, 0xff, 0xfd, 0x19); /* IAC DO EOR */
|
||||||
|
IACSET(init->buf, 0xff, 0xfb, 0x19); /* IAC WILL EOR */
|
||||||
|
IACSET(init->buf, 0xff, 0xfd, 0x00); /* IAC DO BINARY */
|
||||||
|
IACSET(init->buf, 0xff, 0xfb, 0x00); /* IAC WILL BINARY */
|
||||||
|
IACSET(init->buf, 0xff, 0xfd, 0x18); /* IAC DO TERMINAL TYPE */
|
||||||
|
IACSET(init->buf, 0xff, 0xfa, 0x18); /* IAC SB TERMINAL TYPE */
|
||||||
|
IACSET(init->buf, 0x01, 0xff, 0xf0); /* SEND IAC SE */
|
||||||
|
}
|
||||||
|
|
||||||
#undef IACSET
|
#undef IACSET
|
||||||
|
|
||||||
@ -585,7 +615,8 @@ static void tcp_chr_tls_handshake(QIOTask *task,
|
|||||||
if (qio_task_propagate_error(task, NULL)) {
|
if (qio_task_propagate_error(task, NULL)) {
|
||||||
tcp_chr_disconnect(chr);
|
tcp_chr_disconnect(chr);
|
||||||
} else {
|
} else {
|
||||||
if (s->do_telnetopt) {
|
/* tn3270 does not support TLS yet */
|
||||||
|
if (s->do_telnetopt && !s->is_tn3270) {
|
||||||
tcp_chr_telnet_init(chr);
|
tcp_chr_telnet_init(chr);
|
||||||
} else {
|
} else {
|
||||||
tcp_chr_connect(chr);
|
tcp_chr_connect(chr);
|
||||||
@ -824,12 +855,14 @@ static void qmp_chardev_open_socket(Chardev *chr,
|
|||||||
bool do_nodelay = sock->has_nodelay ? sock->nodelay : false;
|
bool do_nodelay = sock->has_nodelay ? sock->nodelay : false;
|
||||||
bool is_listen = sock->has_server ? sock->server : true;
|
bool is_listen = sock->has_server ? sock->server : true;
|
||||||
bool is_telnet = sock->has_telnet ? sock->telnet : false;
|
bool is_telnet = sock->has_telnet ? sock->telnet : false;
|
||||||
|
bool is_tn3270 = sock->has_tn3270 ? sock->tn3270 : false;
|
||||||
bool is_waitconnect = sock->has_wait ? sock->wait : false;
|
bool is_waitconnect = sock->has_wait ? sock->wait : false;
|
||||||
int64_t reconnect = sock->has_reconnect ? sock->reconnect : 0;
|
int64_t reconnect = sock->has_reconnect ? sock->reconnect : 0;
|
||||||
QIOChannelSocket *sioc = NULL;
|
QIOChannelSocket *sioc = NULL;
|
||||||
|
|
||||||
s->is_listen = is_listen;
|
s->is_listen = is_listen;
|
||||||
s->is_telnet = is_telnet;
|
s->is_telnet = is_telnet;
|
||||||
|
s->is_tn3270 = is_tn3270;
|
||||||
s->do_nodelay = do_nodelay;
|
s->do_nodelay = do_nodelay;
|
||||||
if (sock->tls_creds) {
|
if (sock->tls_creds) {
|
||||||
Object *creds;
|
Object *creds;
|
||||||
@ -879,7 +912,7 @@ static void qmp_chardev_open_socket(Chardev *chr,
|
|||||||
addr, is_listen, is_telnet);
|
addr, is_listen, is_telnet);
|
||||||
|
|
||||||
if (is_listen) {
|
if (is_listen) {
|
||||||
if (is_telnet) {
|
if (is_telnet || is_tn3270) {
|
||||||
s->do_telnetopt = 1;
|
s->do_telnetopt = 1;
|
||||||
}
|
}
|
||||||
} else if (reconnect > 0) {
|
} else if (reconnect > 0) {
|
||||||
@ -933,6 +966,7 @@ static void qemu_chr_parse_socket(QemuOpts *opts, ChardevBackend *backend,
|
|||||||
bool is_listen = qemu_opt_get_bool(opts, "server", false);
|
bool is_listen = qemu_opt_get_bool(opts, "server", false);
|
||||||
bool is_waitconnect = is_listen && qemu_opt_get_bool(opts, "wait", true);
|
bool is_waitconnect = is_listen && qemu_opt_get_bool(opts, "wait", true);
|
||||||
bool is_telnet = qemu_opt_get_bool(opts, "telnet", false);
|
bool is_telnet = qemu_opt_get_bool(opts, "telnet", false);
|
||||||
|
bool is_tn3270 = qemu_opt_get_bool(opts, "tn3270", false);
|
||||||
bool do_nodelay = !qemu_opt_get_bool(opts, "delay", true);
|
bool do_nodelay = !qemu_opt_get_bool(opts, "delay", true);
|
||||||
int64_t reconnect = qemu_opt_get_number(opts, "reconnect", 0);
|
int64_t reconnect = qemu_opt_get_number(opts, "reconnect", 0);
|
||||||
const char *path = qemu_opt_get(opts, "path");
|
const char *path = qemu_opt_get(opts, "path");
|
||||||
@ -968,6 +1002,8 @@ static void qemu_chr_parse_socket(QemuOpts *opts, ChardevBackend *backend,
|
|||||||
sock->server = is_listen;
|
sock->server = is_listen;
|
||||||
sock->has_telnet = true;
|
sock->has_telnet = true;
|
||||||
sock->telnet = is_telnet;
|
sock->telnet = is_telnet;
|
||||||
|
sock->has_tn3270 = true;
|
||||||
|
sock->tn3270 = is_tn3270;
|
||||||
sock->has_wait = true;
|
sock->has_wait = true;
|
||||||
sock->wait = is_waitconnect;
|
sock->wait = is_waitconnect;
|
||||||
sock->has_reconnect = true;
|
sock->has_reconnect = true;
|
||||||
|
@ -696,7 +696,8 @@ QemuOpts *qemu_chr_parse_compat(const char *label, const char *filename)
|
|||||||
return opts;
|
return opts;
|
||||||
}
|
}
|
||||||
if (strstart(filename, "tcp:", &p) ||
|
if (strstart(filename, "tcp:", &p) ||
|
||||||
strstart(filename, "telnet:", &p)) {
|
strstart(filename, "telnet:", &p) ||
|
||||||
|
strstart(filename, "tn3270:", &p)) {
|
||||||
if (sscanf(p, "%64[^:]:%32[^,]%n", host, port, &pos) < 2) {
|
if (sscanf(p, "%64[^:]:%32[^,]%n", host, port, &pos) < 2) {
|
||||||
host[0] = 0;
|
host[0] = 0;
|
||||||
if (sscanf(p, ":%32[^,]%n", port, &pos) < 1)
|
if (sscanf(p, ":%32[^,]%n", port, &pos) < 1)
|
||||||
@ -712,8 +713,11 @@ QemuOpts *qemu_chr_parse_compat(const char *label, const char *filename)
|
|||||||
goto fail;
|
goto fail;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
if (strstart(filename, "telnet:", &p))
|
if (strstart(filename, "telnet:", &p)) {
|
||||||
qemu_opt_set(opts, "telnet", "on", &error_abort);
|
qemu_opt_set(opts, "telnet", "on", &error_abort);
|
||||||
|
} else if (strstart(filename, "tn3270:", &p)) {
|
||||||
|
qemu_opt_set(opts, "tn3270", "on", &error_abort);
|
||||||
|
}
|
||||||
return opts;
|
return opts;
|
||||||
}
|
}
|
||||||
if (strstart(filename, "udp:", &p)) {
|
if (strstart(filename, "udp:", &p)) {
|
||||||
@ -1176,6 +1180,9 @@ QemuOptsList qemu_chardev_opts = {
|
|||||||
},{
|
},{
|
||||||
.name = "telnet",
|
.name = "telnet",
|
||||||
.type = QEMU_OPT_BOOL,
|
.type = QEMU_OPT_BOOL,
|
||||||
|
},{
|
||||||
|
.name = "tn3270",
|
||||||
|
.type = QEMU_OPT_BOOL,
|
||||||
},{
|
},{
|
||||||
.name = "tls-creds",
|
.name = "tls-creds",
|
||||||
.type = QEMU_OPT_STRING,
|
.type = QEMU_OPT_STRING,
|
||||||
|
@ -7,6 +7,14 @@
|
|||||||
#include "qemu/bitmap.h"
|
#include "qemu/bitmap.h"
|
||||||
#include "qom/object.h"
|
#include "qom/object.h"
|
||||||
|
|
||||||
|
#define IAC_EOR 239
|
||||||
|
#define IAC_SE 240
|
||||||
|
#define IAC_NOP 241
|
||||||
|
#define IAC_BREAK 243
|
||||||
|
#define IAC_IP 244
|
||||||
|
#define IAC_SB 250
|
||||||
|
#define IAC 255
|
||||||
|
|
||||||
/* character device */
|
/* character device */
|
||||||
|
|
||||||
typedef enum {
|
typedef enum {
|
||||||
|
@ -4877,6 +4877,8 @@
|
|||||||
# @nodelay: set TCP_NODELAY socket option (default: false)
|
# @nodelay: set TCP_NODELAY socket option (default: false)
|
||||||
# @telnet: enable telnet protocol on server
|
# @telnet: enable telnet protocol on server
|
||||||
# sockets (default: false)
|
# sockets (default: false)
|
||||||
|
# @tn3270: enable tn3270 protocol on server
|
||||||
|
# sockets (default: false) (Since: 2.10)
|
||||||
# @reconnect: For a client socket, if a socket is disconnected,
|
# @reconnect: For a client socket, if a socket is disconnected,
|
||||||
# then attempt a reconnect after the given number of seconds.
|
# then attempt a reconnect after the given number of seconds.
|
||||||
# Setting this to zero disables this function. (default: 0)
|
# Setting this to zero disables this function. (default: 0)
|
||||||
@ -4890,6 +4892,7 @@
|
|||||||
'*wait' : 'bool',
|
'*wait' : 'bool',
|
||||||
'*nodelay' : 'bool',
|
'*nodelay' : 'bool',
|
||||||
'*telnet' : 'bool',
|
'*telnet' : 'bool',
|
||||||
|
'*tn3270' : 'bool',
|
||||||
'*reconnect' : 'int' },
|
'*reconnect' : 'int' },
|
||||||
'base': 'ChardevCommon' }
|
'base': 'ChardevCommon' }
|
||||||
|
|
||||||
|
Loading…
Reference in New Issue
Block a user