Files
org-hyperion-cules/comm3705.c
2014-10-27 05:21:50 -07:00

2448 lines
81 KiB
C

/* COMM3705.C (c) Copyright Roger Bowler, 2008-2012 */
/* (c) Copyright MHP, 2007 (see below) */
/* Hercules 3705 communications controller */
/* running NCP */
/* */
/* Released under "The Q Public License Version 1" */
/* (http://www.hercules-390.org/herclic.html) as modifications to */
/* Hercules. */
/***********************************************************************/
/* */
/* comm3705.c - (C) Copyright 2007 by MHP <ikj1234i at yahoo dot com> */
/* */
/* Loosely based on commadpt.c by Ivan Warren */
/* */
/* This module appears to the "host" as a 3705 communications */
/* controller running NCP. It does not attempt to provide an emulated */
/* execution environment for native 3705 code. */
/* */
/* Experimental release 0.02 Oct. 15, 2007 */
/* */
/* A very minimalistic SNA engine is implemented. All received SNA */
/* requests are responded to with a positive response. Also, there's */
/* enough code to enable a single SNA session to logon in LU1 (TTY) */
/* mode. */
/* */
/* FID1 is the only SNA header type supported. */
/* */
/* A large amount of required SNA functionality is not present in this */
/* release. There are no "state machines", "layers", "services", */
/* chaining*, pacing, brackets, etc.etc.etc. There are */
/* probably more bugs than working functions... Enjoy ;-) */
/* */
/* A better implementation might be to separate the SNA functions out */
/* into an independent process, with communications over a full-duplex */
/* TCP/IP socket... We might also get rid of all the magic constants...*/
/* */
/* New in release 0.02 - */
/* - VTAM switched (dial) support */
/* - New remote NCP capability */
/* - SNA 3270 (LU2) support (*with RU chaining) */
/* - fixes for some bugs in 0.01 */
/* New in release 0.03 - */
/* - don't process TTY lines until CR received */
/* New in release 0.04 - */
/* - make debug messages optional */
/* */
/* 73 DE KA1RBI */
/***********************************************************************/
#include "hstdinc.h"
#include "hercules.h"
#include "devtype.h"
#include "opcode.h"
#include "parser.h"
#include "comm3705.h"
#if defined(WIN32) && defined(OPTION_DYNAMIC_LOAD) && !defined(HDL_USE_LIBTOOL) && !defined(_MSVC_)
SYSBLK *psysblk;
#define sysblk (*psysblk)
#endif
#if !defined(min)
#define min(a,b) (((a) <= (b)) ? (a) : (b))
#endif
static void make_sna_requests2(COMMADPT*);
static void make_sna_requests3(COMMADPT*);
static void make_sna_requests4(COMMADPT*, int, BYTE);
static void make_sna_requests5(COMMADPT*);
static unsigned char R010201[3] = {0x01, 0x02, 0x01};
static unsigned char R010202[3] = {0x01, 0x02, 0x02};
static unsigned char R010203[3] = {0x01, 0x02, 0x03};
static unsigned char R010204[3] = {0x01, 0x02, 0x04};
static unsigned char R010205[3] = {0x01, 0x02, 0x05};
static unsigned char R01020A[3] = {0x01, 0x02, 0x0A};
static unsigned char R01020B[3] = {0x01, 0x02, 0x0B};
static unsigned char R01020F[3] = {0x01, 0x02, 0x0F};
static unsigned char R010211[3] = {0x01, 0x02, 0x11};
static unsigned char R010216[3] = {0x01, 0x02, 0x16};
static unsigned char R010217[3] = {0x01, 0x02, 0x17};
static unsigned char R010219[3] = {0x01, 0x02, 0x19};
static unsigned char R01021A[3] = {0x01, 0x02, 0x1A};
static unsigned char R01021B[3] = {0x01, 0x02, 0x1B};
static unsigned char R010280[3] = {0x01, 0x02, 0x80};
static unsigned char R010281[3] = {0x01, 0x02, 0x81};
static unsigned char R010284[3] = {0x01, 0x02, 0x84};
#define BUFPD 0x1C
static BYTE commadpt_immed_command[256]=
{ 0,0,0,1,0,0,0,0,0,0,0,0,0,0,0,0,
0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,
0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,
0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,
0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,
0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,
0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,
0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,
0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,
0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,
0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,
0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,
0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,
0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,
0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,
0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0};
/*---------------------------------------------------------------*/
/* PARSER TABLES */
/*---------------------------------------------------------------*/
static PARSER ptab[]={
{"lport","%s"},
{"lhost","%s"},
{"rport","%s"},
{"rhost","%s"},
{"dial","%s"},
{"rto","%s"},
{"pto","%s"},
{"eto","%s"},
{"switched","%s"},
{"lnctl","%s"},
{"debug","%s"},
{"emu3791","%s"},
{NULL,NULL}
};
enum {
COMMADPT_KW_LPORT=1,
COMMADPT_KW_LHOST,
COMMADPT_KW_RPORT,
COMMADPT_KW_RHOST,
COMMADPT_KW_DIAL,
COMMADPT_KW_READTO,
COMMADPT_KW_POLLTO,
COMMADPT_KW_ENABLETO,
COMMADPT_KW_SWITCHED,
COMMADPT_KW_LNCTL,
COMMADPT_KW_DEBUG,
COMMADPT_KW_EMU3791
} comm3705_kw;
//////////////////////////////////////////////////////////////////////
// some code copied from console.c
static HOST_INFO cons_hostinfo; /* Host info for this system */
/*-------------------------------------------------------------------*/
/* Telnet command definitions */
/*-------------------------------------------------------------------*/
#define BINARY 0 /* Binary Transmission */
#define IS 0 /* Used by terminal-type negotiation */
#define SEND 1 /* Used by terminal-type negotiation */
#define ECHO_OPTION 1 /* Echo option */
#define SUPPRESS_GA 3 /* Suppress go-ahead option */
#define TIMING_MARK 6 /* Timing mark option */
#define TERMINAL_TYPE 24 /* Terminal type option */
#define NAWS 31 /* Negotiate About Window Size */
#define EOR 25 /* End of record option */
#define EOR_MARK 239 /* End of record marker */
#define SE 240 /* End of subnegotiation parameters */
#define NOP 241 /* No operation */
#define DATA_MARK 242 /* The data stream portion of a Synch.
This should always be accompanied
by a TCP Urgent notification */
#define BRK 243 /* Break character */
#define IP 244 /* Interrupt Process */
#define AO 245 /* Abort Output */
#define AYT 246 /* Are You There */
#define EC 247 /* Erase character */
#define EL 248 /* Erase Line */
#define GA 249 /* Go ahead */
#define SB 250 /* Subnegotiation of indicated option */
#define WILL 251 /* Indicates the desire to begin
performing, or confirmation that
you are now performing, the
indicated option */
#define WONT 252 /* Indicates the refusal to perform,
or continue performing, the
indicated option */
#define DO 253 /* Indicates the request that the
other party perform, or
confirmation that you are expecting
the other party to perform, the
indicated option */
#define DONT 254 /* Indicates the demand that the
other party stop performing,
or confirmation that you are no
longer expecting the other party
to perform, the indicated option */
#define IAC 255 /* Interpret as Command */
/*-------------------------------------------------------------------*/
/* 3270 definitions */
/*-------------------------------------------------------------------*/
/* 3270 remote commands */
#define R3270_EAU 0x6F /* Erase All Unprotected */
#define R3270_EW 0xF5 /* Erase/Write */
#define R3270_EWA 0x7E /* Erase/Write Alternate */
#define R3270_RB 0xF2 /* Read Buffer */
#define R3270_RM 0xF6 /* Read Modified */
#define R3270_RMA 0x6E /* Read Modified All */
#define R3270_WRT 0xF1 /* Write */
#define R3270_WSF 0xF3 /* Write Structured Field */
/* 3270 orders */
#define O3270_SBA 0x11 /* Set Buffer Address */
#define O3270_SF 0x1D /* Start Field */
#define O3270_SFE 0x29 /* Start Field Extended */
#define O3270_SA 0x28 /* Set Attribute */
#define O3270_IC 0x13 /* Insert Cursor */
#define O3270_MF 0x2C /* Modify Field */
#define O3270_PT 0x05 /* Program Tab */
#define O3270_RA 0x3C /* Repeat to Address */
#define O3270_EUA 0x12 /* Erase Unprotected to Addr */
#define O3270_GE 0x08 /* Graphic Escape */
/* Inbound structured fields */
#define SF3270_AID 0x88 /* Aid value of inbound SF */
#define SF3270_3270DS 0x80 /* SFID of 3270 datastream SF*/
/*-------------------------------------------------------------------*/
/* Internal macro definitions */
/*-------------------------------------------------------------------*/
/* DEBUG_LVL: 0 = none
1 = status
2 = headers
3 = buffers
*/
#define DEBUG_LVL 0
#if DEBUG_LVL == 0
#define TNSDEBUG1 1 ? ((void)0) : logmsg
#define TNSDEBUG2 1 ? ((void)0) : logmsg
#define TNSDEBUG3 1 ? ((void)0) : logmsg
#endif
#if DEBUG_LVL == 1
#define TNSDEBUG1 logmsg
#define TNSDEBUG2 1 ? ((void)0) : logmsg
#define TNSDEBUG3 1 ? ((void)0) : logmsg
#endif
#if DEBUG_LVL == 2
#define TNSDEBUG1 logmsg
#define TNSDEBUG2 logmsg
#define TNSDEBUG3 1 ? ((void)0) : logmsg
#endif
#if DEBUG_LVL == 3
#define TNSDEBUG1 logmsg
#define TNSDEBUG2 logmsg
#define TNSDEBUG3 logmsg
#endif
#define BUFLEN_3270 65536 /* 3270 Send/Receive buffer */
#define BUFLEN_1052 150 /* 1052 Send/Receive buffer */
#undef FIX_QWS_BUG_FOR_MCS_CONSOLES
/*-------------------------------------------------------------------*/
/* SUBROUTINE TO TRACE THE CONTENTS OF AN ASCII MESSAGE PACKET */
/*-------------------------------------------------------------------*/
#if DEBUG_LVL == 3
static void
packet_trace( BYTE* pAddr, int iLen )
{
int offset;
unsigned int i;
u_char c = '\0';
u_char e = '\0';
u_char print_ascii[17];
u_char print_ebcdic[17];
u_char print_line[64];
u_char tmp[32];
for( offset = 0; offset < iLen; )
{
memset( print_ascii, 0, sizeof( print_ascii ) );
memset( print_ebcdic, 0, sizeof( print_ebcdic ) );
memset( print_line, 0, sizeof( print_line ) );
sprintf(print_line, "+%4.4X ", offset );
for( i = 0; i < 16; i++ )
{
c = *pAddr++;
if( offset < iLen )
{
sprintf( tmp, "%2.2X", c );
strlcat( print_line, tmp, sizeof( print_line ));
print_ebcdic[i] = print_ascii[i] = '.';
e = guest_to_host( c );
if( isprint( e ) )
print_ebcdic[i] = e;
if( isprint( c ) )
print_ascii[i] = c;
}
else
{
strlcat( print_line, " ", sizeof( print_line ));
}
offset++;
if( ( offset & 3 ) == 0 )
{
strlcat( print_line, " ", sizeof( print_line ));
}
}
logmsg( "%s %s %s\n", print_line, print_ascii, print_ebcdic );
}
} /* end function packet_trace */
#else
#define packet_trace( _addr, _len)
#endif
#if 0
static struct sockaddr_in * get_inet_socket(char *host_serv)
{
char *host = NULL;
char *serv;
struct sockaddr_in *sin;
if((serv = strchr(host_serv,':')))
{
*serv++ = '\0';
if(*host_serv)
host = host_serv;
}
else
serv = host_serv;
if(!(sin = malloc(sizeof(struct sockaddr_in))))
return sin;
sin->sin_family = AF_INET;
if(host)
{
struct hostent *hostent;
hostent = gethostbyname(host);
if(!hostent)
{
WRMSG(HHC01016, "I", "IP address", host);
free(sin);
return NULL;
}
memcpy(&sin->sin_addr,*hostent->h_addr_list,sizeof(sin->sin_addr));
}
else
sin->sin_addr.s_addr = INADDR_ANY;
if(serv)
{
if(!isdigit(*serv))
{
struct servent *servent;
servent = getservbyname(serv, "tcp");
if(!servent)
{
WRMSG(HHC01016, "I", "port number", host);
free(sin);
return NULL;
}
sin->sin_port = servent->s_port;
}
else
sin->sin_port = htons(atoi(serv));
}
else
{
WRMSG(HHC01017, "E", host_serv);
free(sin);
return NULL;
}
return sin;
}
#endif
#if FALSE
/*-------------------------------------------------------------------*/
/* SUBROUTINE TO REMOVE ANY IAC SEQUENCES FROM THE DATA STREAM */
/* Returns the new length after deleting IAC commands */
/*-------------------------------------------------------------------*/
static int
remove_iac (BYTE *buf, int len)
{
int m, n, c;
for (m=0, n=0; m < len; ) {
/* Interpret IAC commands */
if (buf[m] == IAC) {
/* Treat IAC in last byte of buffer as IAC NOP */
c = (++m < len)? buf[m++] : NOP;
/* Process IAC command */
switch (c) {
case IAC: /* Insert single IAC in buffer */
buf[n++] = IAC;
break;
case BRK: /* Set ATTN indicator */
break;
case IP: /* Set SYSREQ indicator */
break;
case WILL: /* Skip option negotiation command */
case WONT:
case DO:
case DONT:
m++;
break;
case SB: /* Skip until IAC SE sequence found */
for (; m < len; m++) {
if (buf[m] != IAC) continue;
if (++m >= len) break;
if (buf[m] == SE) { m++; break; }
} /* end for */
default: /* Ignore NOP or unknown command */
break;
} /* end switch(c) */
} else {
/* Copy data bytes */
if (n < m) buf[n] = buf[m];
m++; n++;
}
} /* end for */
if (n < m) {
TNSDEBUG3("console: DBG001: %d IAC bytes removed, newlen=%d\n",
m-n, n);
packet_trace (buf, n);
}
return n;
} /* end function remove_iac */
#endif
/*-------------------------------------------------------------------*/
/* SUBROUTINE TO DOUBLE UP ANY IAC BYTES IN THE DATA STREAM */
/* Returns the new length after inserting extra IAC bytes */
/*-------------------------------------------------------------------*/
static int
double_up_iac (BYTE *buf, int len)
{
int m, n, x, newlen;
/* Count the number of IAC bytes in the data */
for (x=0, n=0; n < len; n++)
if (buf[n] == IAC) x++;
/* Exit if nothing to do */
if (x == 0) return len;
/* Insert extra IAC bytes backwards from the end of the buffer */
newlen = len + x;
TNSDEBUG3("console: DBG002: %d IAC bytes added, newlen=%d\n",
x, newlen);
for (n=newlen, m=len; n > m; ) {
buf[--n] = buf[--m];
if (buf[n] == IAC) buf[--n] = IAC;
}
packet_trace (buf, newlen);
return newlen;
} /* end function double_up_iac */
/*-------------------------------------------------------------------*/
/* SUBROUTINE TO TRANSLATE A NULL-TERMINATED STRING TO EBCDIC */
/*-------------------------------------------------------------------*/
static BYTE *
translate_to_ebcdic (char *str)
{
int i; /* Array subscript */
BYTE c; /* Character work area */
for (i = 0; str[i] != '\0'; i++)
{
c = str[i];
str[i] = (isprint(c) ? host_to_guest(c) : SPACE);
}
return (BYTE *)str;
} /* end function translate_to_ebcdic */
/*-------------------------------------------------------------------*/
/* SUBROUTINE TO SEND A DATA PACKET TO THE CLIENT */
/*-------------------------------------------------------------------*/
static int
send_packet (int csock, BYTE *buf, int len, char *caption)
{
int rc; /* Return code */
if (caption != NULL) {
TNSDEBUG2("console: DBG003: Sending %s\n", caption);
packet_trace (buf, len);
}
rc = send (csock, buf, len, 0);
if (rc < 0) {
WRMSG(HHC01034, "E", "send()", strerror(HSO_errno));
return -1;
} /* end if(rc) */
return 0;
} /* end function send_packet */
/*-------------------------------------------------------------------*/
/* SUBROUTINE TO RECEIVE A DATA PACKET FROM THE CLIENT */
/* This subroutine receives bytes from the client. It stops when */
/* the receive buffer is full, or when the last two bytes received */
/* consist of the IAC character followed by a specified delimiter. */
/* If zero bytes are received, this means the client has closed the */
/* connection, and this is treated as an error. */
/* Input: */
/* csock is the socket number */
/* buf points to area to receive data */
/* reqlen is the number of bytes requested */
/* delim is the delimiter character (0=no delimiter) */
/* Output: */
/* buf is updated with data received */
/* The return value is the number of bytes received, or */
/* -1 if an error occurred. */
/*-------------------------------------------------------------------*/
static int
recv_packet (int csock, BYTE *buf, int reqlen, BYTE delim)
{
int rc=0; /* Return code */
int rcvlen=0; /* Length of data received */
while (rcvlen < reqlen) {
rc = recv (csock, buf + rcvlen, reqlen - rcvlen, 0);
if (rc < 0) {
WRMSG(HHC01034, "E", "recv()", strerror(HSO_errno));
return -1;
}
if (rc == 0) {
TNSDEBUG1("console: DBG004: Connection closed by client\n");
return -1;
}
rcvlen += rc;
if (delim != '\0' && rcvlen >= 2
&& buf[rcvlen-2] == IAC && buf[rcvlen-1] == delim)
break;
}
TNSDEBUG2("console: DBG005: Packet received length=%d\n", rcvlen);
packet_trace (buf, rcvlen);
return rcvlen;
} /* end function recv_packet */
/*-------------------------------------------------------------------*/
/* SUBROUTINE TO RECEIVE A PACKET AND COMPARE WITH EXPECTED VALUE */
/*-------------------------------------------------------------------*/
static int
expect (int csock, BYTE *expected, int len, char *caption)
{
int rc; /* Return code */
BYTE buf[512]; /* Receive buffer */
#if 1
/* TCP/IP for MVS returns the server sequence rather then
the client sequence during bin negotiation 19/06/00 Jan Jaeger */
static BYTE do_bin[] = { IAC, DO, BINARY, IAC, WILL, BINARY };
static BYTE will_bin[] = { IAC, WILL, BINARY, IAC, DO, BINARY };
#endif
UNREFERENCED(caption);
rc = recv_packet (csock, buf, len, 0);
if (rc < 0)
return -1;
#if 1
/* TCP/IP FOR MVS DOES NOT COMPLY TO RFC 1576 THIS IS A BYPASS */
if(memcmp(buf, expected, len) != 0
&& !(len == sizeof(will_bin)
&& memcmp(expected, will_bin, len) == 0
&& memcmp(buf, do_bin, len) == 0) )
#else
if (memcmp(buf, expected, len) != 0)
#endif
{
TNSDEBUG2("console: DBG006: Expected %s\n", caption);
return -1;
}
TNSDEBUG2("console: DBG007: Received %s\n", caption);
return 0;
} /* end function expect */
/*-------------------------------------------------------------------*/
/* SUBROUTINE TO NEGOTIATE TELNET PARAMETERS */
/* This subroutine negotiates the terminal type with the client */
/* and uses the terminal type to determine whether the client */
/* is to be supported as a 3270 display console or as a 1052/3215 */
/* printer-keyboard console. */
/* */
/* Valid display terminal types are "IBM-NNNN", "IBM-NNNN-M", and */
/* "IBM-NNNN-M-E", where NNNN is 3270, 3277, 3278, 3279, 3178, 3179, */
/* or 3180, M indicates the screen size (2=25x80, 3=32x80, 4=43x80, */
/* 5=27x132, X=determined by Read Partition Query command), and */
/* -E is an optional suffix indicating that the terminal supports */
/* extended attributes. Displays are negotiated into tn3270 mode. */
/* An optional device number suffix (example: IBM-3270@01F) may */
/* be specified to request allocation to a specific device number. */
/* Valid 3270 printer type is "IBM-3287-1" */
/* */
/* Terminal types whose first four characters are not "IBM-" are */
/* handled as printer-keyboard consoles using telnet line mode. */
/* */
/* Input: */
/* csock Socket number for client connection */
/* Output: */
/* class D=3270 display console, K=printer-keyboard console */
/* P=3270 printer */
/* model 3270 model indicator (2,3,4,5,X) */
/* extatr 3270 extended attributes (Y,N) */
/* devn Requested device number, or FFFF=any device number */
/* Return value: */
/* 0=negotiation successful, -1=negotiation error */
/*-------------------------------------------------------------------*/
static int
negotiate(int csock, BYTE *class, BYTE *model, BYTE *extatr, U16 *devn,char *group)
{
int rc; /* Return code */
char *termtype; /* Pointer to terminal type */
char *s; /* String pointer */
BYTE c; /* Trailing character */
U16 devnum; /* Requested device number */
BYTE buf[512]; /* Telnet negotiation buffer */
static BYTE do_term[] = { IAC, DO, TERMINAL_TYPE };
static BYTE will_term[] = { IAC, WILL, TERMINAL_TYPE };
static BYTE req_type[] = { IAC, SB, TERMINAL_TYPE, SEND, IAC, SE };
static BYTE type_is[] = { IAC, SB, TERMINAL_TYPE, IS };
static BYTE do_eor[] = { IAC, DO, EOR, IAC, WILL, EOR };
static BYTE will_eor[] = { IAC, WILL, EOR, IAC, DO, EOR };
static BYTE do_bin[] = { IAC, DO, BINARY, IAC, WILL, BINARY };
static BYTE will_bin[] = { IAC, WILL, BINARY, IAC, DO, BINARY };
#if 0
static BYTE do_tmark[] = { IAC, DO, TIMING_MARK };
static BYTE will_tmark[] = { IAC, WILL, TIMING_MARK };
static BYTE wont_sga[] = { IAC, WONT, SUPPRESS_GA };
static BYTE dont_sga[] = { IAC, DONT, SUPPRESS_GA };
#endif
static BYTE wont_echo[] = { IAC, WONT, ECHO_OPTION };
static BYTE dont_echo[] = { IAC, DONT, ECHO_OPTION };
static BYTE will_naws[] = { IAC, WILL, NAWS };
/* Perform terminal-type negotiation */
rc = send_packet (csock, do_term, sizeof(do_term),
"IAC DO TERMINAL_TYPE");
if (rc < 0) return -1;
rc = expect (csock, will_term, sizeof(will_term),
"IAC WILL TERMINAL_TYPE");
if (rc < 0) return -1;
/* Request terminal type */
rc = send_packet (csock, req_type, sizeof(req_type),
"IAC SB TERMINAL_TYPE SEND IAC SE");
if (rc < 0) return -1;
rc = recv_packet (csock, buf, sizeof(buf)-2, SE);
if (rc < 0) return -1;
/* Ignore Negotiate About Window Size */
if (rc >= (int)sizeof(will_naws) &&
memcmp (buf, will_naws, sizeof(will_naws)) == 0)
{
memmove(buf, &buf[sizeof(will_naws)], (rc - sizeof(will_naws)));
rc -= sizeof(will_naws);
}
if (rc < (int)(sizeof(type_is) + 2)
|| memcmp(buf, type_is, sizeof(type_is)) != 0
|| buf[rc-2] != IAC || buf[rc-1] != SE) {
TNSDEBUG2("console: DBG008: Expected IAC SB TERMINAL_TYPE IS\n");
return -1;
}
buf[rc-2] = '\0';
termtype = (char *)(buf + sizeof(type_is));
TNSDEBUG2("console: DBG009: Received IAC SB TERMINAL_TYPE IS %s IAC SE\n",
termtype);
/* Check terminal type string for device name suffix */
s = strchr (termtype, '@');
if(s!=NULL)
{
if(strlen(s)<16)
{
strlcpy(group,&s[1],16);
}
}
else
{
group[0]=0;
}
if (s != NULL && sscanf (s, "@%hx%c", &devnum,&c) == 1)
{
*devn = devnum;
group[0]=0;
}
else
{
*devn = 0xFFFF;
}
/* Test for non-display terminal type */
if (memcmp(termtype, "IBM-", 4) != 0)
{
#if 0
/* Perform line mode negotiation */
rc = send_packet (csock, do_tmark, sizeof(do_tmark),
"IAC DO TIMING_MARK");
if (rc < 0) return -1;
rc = expect (csock, will_tmark, sizeof(will_tmark),
"IAC WILL TIMING_MARK");
if (rc < 0) return 0;
rc = send_packet (csock, wont_sga, sizeof(wont_sga),
"IAC WONT SUPPRESS_GA");
if (rc < 0) return -1;
rc = expect (csock, dont_sga, sizeof(dont_sga),
"IAC DONT SUPPRESS_GA");
if (rc < 0) return -1;
#endif
if (memcmp(termtype, "ANSI", 4) == 0)
{
rc = send_packet (csock, wont_echo, sizeof(wont_echo),
"IAC WONT ECHO");
if (rc < 0) return -1;
rc = expect (csock, dont_echo, sizeof(dont_echo),
"IAC DONT ECHO");
if (rc < 0) return -1;
}
/* Return printer-keyboard terminal class */
*class = 'K';
*model = '-';
*extatr = '-';
return 0;
}
/* Determine display terminal model */
if (memcmp(termtype+4,"DYNAMIC",7) == 0) {
*model = 'X';
*extatr = 'Y';
} else {
if (!(memcmp(termtype+4, "3277", 4) == 0
|| memcmp(termtype+4, "3270", 4) == 0
|| memcmp(termtype+4, "3178", 4) == 0
|| memcmp(termtype+4, "3278", 4) == 0
|| memcmp(termtype+4, "3179", 4) == 0
|| memcmp(termtype+4, "3180", 4) == 0
|| memcmp(termtype+4, "3287", 4) == 0
|| memcmp(termtype+4, "3279", 4) == 0))
return -1;
*model = '2';
*extatr = 'N';
if (termtype[8]=='-') {
if (termtype[9] < '1' || termtype[9] > '5')
return -1;
*model = termtype[9];
if (memcmp(termtype+4, "328",3) == 0) *model = '2';
if (memcmp(termtype+10, "-E", 2) == 0)
*extatr = 'Y';
}
}
/* Perform end-of-record negotiation */
rc = send_packet (csock, do_eor, sizeof(do_eor),
"IAC DO EOR IAC WILL EOR");
if (rc < 0) return -1;
rc = expect (csock, will_eor, sizeof(will_eor),
"IAC WILL EOR IAC DO EOR");
if (rc < 0) return -1;
/* Perform binary negotiation */
rc = send_packet (csock, do_bin, sizeof(do_bin),
"IAC DO BINARY IAC WILL BINARY");
if (rc < 0) return -1;
rc = expect (csock, will_bin, sizeof(will_bin),
"IAC WILL BINARY IAC DO BINARY");
if (rc < 0) return -1;
/* Return display terminal class */
if (memcmp(termtype+4,"3287",4)==0) *class='P';
else *class = 'D';
return 0;
} /* end function negotiate */
/*-------------------------------------------------------------------*/
/* NEW CLIENT CONNECTION */
/*-------------------------------------------------------------------*/
static int
connect_client (int *csockp)
/* returns 1 if 3270, else 0 */
{
int rc; /* Return code */
size_t len; /* Data length */
int csock; /* Socket for conversation */
struct sockaddr_in client; /* Client address structure */
socklen_t namelen; /* Length of client structure*/
char *clientip; /* Addr of client ip address */
U16 devnum; /* Requested device number */
BYTE class; /* D=3270, P=3287, K=3215/1052 */
BYTE model; /* 3270 model (2,3,4,5,X) */
BYTE extended; /* Extended attributes (Y,N) */
char buf[256]; /* Message buffer */
char conmsg[256]; /* Connection message */
char devmsg[25]; /* Device message */
char hostmsg[256]; /* Host ID message */
char num_procs[16]; /* #of processors string */
char group[16]; /* Console group */
/* Load the socket address from the thread parameter */
csock = *csockp;
/* Obtain the client's IP address */
namelen = sizeof(client);
rc = getpeername (csock, (struct sockaddr *)&client, &namelen);
/* Log the client's IP address and hostname */
clientip = strdup(inet_ntoa(client.sin_addr));
#if 0
// The following isn't really needed and hangs under unusual
// network configuration settings and thus has been removed.
{
struct hostent* pHE; /* Addr of hostent structure */
char* clientname; /* Addr of client hostname */
pHE = gethostbyaddr ((unsigned char*)(&client.sin_addr),
sizeof(client.sin_addr), AF_INET);
if (pHE != NULL && pHE->h_name != NULL
&& pHE->h_name[0] != '\0') {
clientname = (char*) pHE->h_name;
} else {
clientname = "host name unknown";
}
TNSDEBUG1("console: DBG018: Received connection from %s (%s)\n",
clientip, clientname);
}
#else
TNSDEBUG1("console: DBG018: Received connection from %s\n",
clientip );
#endif
/* Negotiate telnet parameters */
rc = negotiate (csock, &class, &model, &extended, &devnum, group);
if (rc != 0)
{
close_socket (csock);
if (clientip) free(clientip);
return 0;
}
/* Build connection message for client */
if ( cons_hostinfo.num_procs > 1 )
snprintf( num_procs, sizeof(num_procs)-1, "MP=%d", cons_hostinfo.num_procs );
else
strlcpy( num_procs, "UP", sizeof(num_procs) );
snprintf
(
hostmsg, sizeof(hostmsg),
"running on %s (%s-%s.%s %s %s)"
,cons_hostinfo.nodename
,cons_hostinfo.sysname
,cons_hostinfo.release
,cons_hostinfo.version
,cons_hostinfo.machine
,num_procs
);
snprintf (conmsg, sizeof(conmsg)-1,
"Hercules version %s built on %s %s",
VERSION, __DATE__, __TIME__);
{
snprintf (devmsg, sizeof(devmsg)-1, "Connected to device %4.4X",
0);
}
WRMSG(HHC01018, "I", 0, 0, clientip, 0x3270);
/* Send connection message to client */
if (class != 'K')
{
len = snprintf (buf, sizeof(buf)-1,
"\xF5\x40\x11\x40\x40\x1D\x60%s"
"\x11\xC1\x50\x1D\x60%s"
"\x11\xC2\x60\x1D\x60%s",
translate_to_ebcdic(conmsg),
translate_to_ebcdic(hostmsg),
translate_to_ebcdic(devmsg));
if (len < sizeof(buf))
{
buf[len++] = IAC;
}
else
{
ASSERT(FALSE);
}
if (len < sizeof(buf))
{
buf[len++] = EOR_MARK;
}
else
{
ASSERT(FALSE);
}
}
else
{
len = snprintf (buf, sizeof(buf)-1, "%s\r\n%s\r\n%s\r\n",
conmsg, hostmsg, devmsg);
}
if (class != 'P') /* do not write connection resp on 3287 */
{
rc = send_packet (csock, (BYTE *)buf, (int)len, "CONNECTION RESPONSE");
}
return (class == 'D') ? 1 : 0; /* return 1 if 3270 */
} /* end function connect_client */
static void logdump(char *txt,DEVBLK *dev,BYTE *bfr,size_t sz)
{
char buf[128];
char byte[5];
size_t i;
if(!dev->ccwtrace)
{
return;
}
WRMSG(HHC01048,"D",SSID_TO_LCSS(dev->ssid),dev->devnum,txt);
WRMSG(HHC01049,"D",SSID_TO_LCSS(dev->ssid),dev->devnum,txt,(u_int)sz,(u_int)sz);
buf[0] = 0;
for(i=0;i<sz;i++)
{
if(i%16==0)
{
if(i!=0)
{
WRMSG(HHC01050,"D",SSID_TO_LCSS(dev->ssid),dev->devnum,txt,buf);
buf[0] = 0;
}
MSGBUF(buf, ": %04X:", (unsigned) i);
}
if(i%4==0 && i)
{
strlcat(buf, " ", sizeof(buf));
}
MSGBUF(byte, "%02X", bfr[i]);
strlcat(buf, byte, sizeof(buf));
}
WRMSG(HHC01050,"D",SSID_TO_LCSS(dev->ssid),dev->devnum,txt,buf);
buf[0] = 0;
for(i=0;i<sz;i++)
{
if(i%16==0)
{
if(i!=0)
{
WRMSG(HHC01051,"D",SSID_TO_LCSS(dev->ssid),dev->devnum,buf);
buf[0] = 0;
}
}
MSGBUF(byte, "%c",(bfr[i] & 0x7f) < 0x20 ? '.' : (bfr[i] & 0x7f));
strlcat(buf, byte, sizeof(buf));
}
WRMSG(HHC01051,"D",SSID_TO_LCSS(dev->ssid),dev->devnum,buf);
}
static void put_bufpool(void ** anchor, BYTE * ele) {
void ** elep = anchor;
for (;;) {
if (!*elep) break;
elep = *elep;
}
*elep = ele;
*(void**)ele = 0;
}
static BYTE * get_bufpool(void ** anchor) {
void ** elep = *anchor;
if (elep) {
*anchor = *elep;
} else {
*anchor = 0;
}
return (BYTE*)elep;
}
static void init_bufpool(COMMADPT *ca) {
BYTE * areap;
int i1;
int numbufs = 64;
int bufsize = 256+16+4;
ca->poolarea = (BYTE*)calloc (numbufs, bufsize);
if (!ca->poolarea) {
return;
}
areap = ca->poolarea;
for (i1 = 0; i1 < numbufs; i1++) {
put_bufpool(&ca->freeq, areap);
areap += (bufsize);
}
}
static void free_bufpool(COMMADPT *ca) {
ca->sendq = 0;
ca->freeq = 0;
if (ca->poolarea) {
free(ca->poolarea);
ca->poolarea = 0;
}
}
/*-------------------------------------------------------------------*/
/* Free all private structures and buffers */
/*-------------------------------------------------------------------*/
static void commadpt_clean_device(DEVBLK *dev)
{
if(dev->commadpt!=NULL)
{
free(dev->commadpt);
dev->commadpt=NULL;
if(dev->ccwtrace)
{
WRMSG(HHC01052,"D",
SSID_TO_LCSS(dev->ssid),
dev->devnum,"control block freed");
}
}
else
{
if(dev->ccwtrace)
{
WRMSG(HHC01052,"D",
SSID_TO_LCSS(dev->ssid),
dev->devnum,"control block not freed: not allocated");
}
}
return;
}
/*-------------------------------------------------------------------*/
/* Allocate initial private structures */
/*-------------------------------------------------------------------*/
static int commadpt_alloc_device(DEVBLK *dev)
{
dev->commadpt=malloc(sizeof(COMMADPT));
if(dev->commadpt==NULL)
{
char buf[40];
MSGBUF(buf, "malloc(%d)", (int)sizeof(COMMADPT));
WRMSG(HHC01000, "E", SSID_TO_LCSS(dev->ssid), dev->devnum, buf, strerror(errno));
return -1;
}
memset(dev->commadpt,0,sizeof(COMMADPT));
dev->commadpt->dev=dev;
return 0;
}
/*-------------------------------------------------------------------*/
/* Parsing utilities */
/*-------------------------------------------------------------------*/
/*-------------------------------------------------------------------*/
/* commadpt_getport : returns a port number or -1 */
/*-------------------------------------------------------------------*/
static int commadpt_getport(char *txt)
{
int pno;
struct servent *se;
pno=atoi(txt);
if(pno==0)
{
se=getservbyname(txt,"tcp");
if(se==NULL)
{
return -1;
}
pno=se->s_port;
}
return(pno);
}
/*-------------------------------------------------------------------*/
/* commadpt_getaddr : set an in_addr_t if ok, else return -1 */
/*-------------------------------------------------------------------*/
static int commadpt_getaddr(in_addr_t *ia,char *txt)
{
struct hostent *he;
he=gethostbyname(txt);
if(he==NULL)
{
return(-1);
}
memcpy(ia,he->h_addr_list[0],4);
return(0);
}
static void connect_message(int sfd, int na, int flag) {
struct sockaddr_in client;
socklen_t namelen;
char *ipaddr;
char msgtext[256];
if (!sfd)
return;
namelen = sizeof(client);
(void)getpeername (sfd, (struct sockaddr *)&client, &namelen);
ipaddr = inet_ntoa(client.sin_addr);
if (flag == 0)
MSGBUF( msgtext, "%s:%d VTAM CONNECTION ACCEPTED - NETWORK NODE= %4.4X",
ipaddr, (int)ntohs(client.sin_port), na);
else
MSGBUF( msgtext, "%s:%d VTAM CONNECTION TERMINATED",
ipaddr, (int)ntohs(client.sin_port));
WRMSG( HHC01047, "I", msgtext );
write(sfd, msgtext, (u_int)strlen(msgtext));
write(sfd, "\r\n", 2);
}
static void commadpt_read_tty(COMMADPT *ca, BYTE * bfr, int len)
// everything has been tortured to now do 3270 also
{
BYTE bfr3[3];
BYTE c;
int i1;
int eor=0;
logdump("RECV",ca->dev, bfr,len);
/* If there is a complete data record already in the buffer
then discard it before reading more data
For TTY, allow data to accumulate until CR is received */
if (ca->is_3270) {
if (ca->inpbufl) {
ca->rlen3270 = 0;
ca->inpbufl = 0;
}
}
for (i1 = 0; i1 < len; i1++) {
c = (unsigned char) bfr[i1];
if (ca->telnet_opt) {
ca->telnet_opt = 0;
if(ca->dev->ccwtrace)
WRMSG(HHC01053,"D",
SSID_TO_LCSS(ca->dev->ssid), ca->dev->devnum,
ca->telnet_cmd, c);
bfr3[0] = 0xff; /* IAC */
/* set won't/don't for all received commands */
bfr3[1] = (ca->telnet_cmd == 0xfd) ? 0xfc : 0xfe;
bfr3[2] = c;
if (ca->sfd > 0) {
write_socket(ca->sfd,bfr3,3);
}
if(ca->dev->ccwtrace)
WRMSG(HHC01054,"D",
SSID_TO_LCSS(ca->dev->ssid), ca->dev->devnum,
bfr3[1], bfr3[2]);
continue;
}
if (ca->telnet_iac) {
ca->telnet_iac = 0;
if(ca->dev->ccwtrace)
WRMSG(HHC01055, "D",
SSID_TO_LCSS(ca->dev->ssid), ca->dev->devnum,
c);
switch (c) {
case 0xFB: /* TELNET WILL option cmd */
case 0xFD: /* TELNET DO option cmd */
ca->telnet_opt = 1;
ca->telnet_cmd = c;
break;
case 0xF4: /* TELNET interrupt */
if (!ca->telnet_int) {
ca->telnet_int = 1;
}
break;
case EOR_MARK:
eor = 1;
break;
case 0xFF: /* IAC IAC */
ca->inpbuf[ca->rlen3270++] = 0xFF;
break;
}
continue;
}
if (c == 0xFF) { /* TELNET IAC */
ca->telnet_iac = 1;
continue;
} else {
ca->telnet_iac = 0;
}
if (!ca->is_3270) {
if (c == 0x0D) // CR in TTY mode ?
ca->eol_flag = 1;
c = host_to_guest(c); // translate ASCII to EBCDIC for tty
}
ca->inpbuf[ca->rlen3270++] = c;
}
/* received data (rlen3270 > 0) is sufficient for 3270,
but for TTY, eol_flag must also be set */
if ((ca->eol_flag || ca->is_3270) && ca->rlen3270)
{
ca->eol_flag = 0;
if (ca->is_3270)
{
if (eor)
{
ca->inpbufl = ca->rlen3270;
ca->rlen3270 = 0; /* for next msg */
}
}
else
{
ca->inpbufl = ca->rlen3270;
ca->rlen3270 = 0; /* for next msg */
}
if(ca->dev->ccwtrace)
WRMSG(HHC01056, "D",
SSID_TO_LCSS(ca->dev->ssid),
ca->dev->devnum,
ca->inpbufl);
}
}
static void *telnet_thread(void *vca)
{
COMMADPT *ca;
int devnum; /* device number copy for convenience*/
int sockopt; /* Used for setsocketoption */
int rc; /* return code from various rtns */
struct sockaddr_in sin; /* bind socket address structure */
BYTE bfr[256];
ca=(COMMADPT*)vca;
/* get a work copy of devnum (for messages) */
ca->sfd = 0;
devnum=ca->devnum;
ca->lfd=socket(AF_INET,SOCK_STREAM,0);
if(!socket_is_socket(ca->lfd))
{
WRMSG(HHC01002, "E",SSID_TO_LCSS(ca->dev->ssid),devnum,strerror(HSO_errno));
ca->have_cthread=0;
release_lock(&ca->lock);
return NULL;
}
/* Reuse the address regardless of any */
/* spurious connection on that port */
sockopt=1;
setsockopt(ca->lfd,SOL_SOCKET,SO_REUSEADDR,(GETSET_SOCKOPT_T*)&sockopt,sizeof(sockopt));
/* Bind the socket */
sin.sin_family=AF_INET;
sin.sin_addr.s_addr=ca->lhost;
sin.sin_port=htons(ca->lport);
rc=bind(ca->lfd,(struct sockaddr *)&sin,sizeof(sin));
if (rc < 0)
{
WRMSG(HHC01000, "E",SSID_TO_LCSS(ca->dev->ssid),devnum,"bind()",strerror(HSO_errno));
return NULL;
}
/* Start the listen */
rc = listen(ca->lfd,10);
if (rc < 0)
{
WRMSG(HHC01000, "E",SSID_TO_LCSS(ca->dev->ssid),devnum,"listen()",strerror(HSO_errno));
return NULL;
}
WRMSG(HHC01004, "I", SSID_TO_LCSS(ca->dev->ssid), devnum, ca->lport);
for (;;)
{
ca->sfd = 0;
ca->sfd=accept(ca->lfd,NULL,0);
if (ca->sfd < 1)
continue;
if (connect_client(&ca->sfd))
{
ca->is_3270 = 1;
}
else
{
ca->is_3270 = 0;
}
socket_set_blocking_mode(ca->sfd,0); // set to non-blocking mode
make_sna_requests4(ca, 0, (ca->is_3270) ? 0x02 : 0x01); // send REQCONT
ca->hangup = 0;
for (;;)
{
usleep(50000);
if (ca->hangup)
break;
/* read_socket has changed from 3.04 to 3.06 - we need old way */
#ifdef _MSVC_
rc=recv(ca->sfd,bfr,256-BUFPD,0);
#else
rc=read(ca->sfd,bfr,256-BUFPD);
#endif
if (rc < 0)
{
if (0
#ifndef WIN32
|| EAGAIN == errno
#endif
|| HSO_EWOULDBLOCK == HSO_errno
)
{
continue;
}
break;
}
if (rc == 0)
{
// make_sna_requests4(ca, 1); // send REQDISCONT
make_sna_requests5(ca);
break;
}
commadpt_read_tty(ca,bfr,rc);
}
close_socket(ca->sfd);
ca->sfd = 0;
}
UNREACHABLE_CODE();
}
/*-------------------------------------------------------------------*/
/* Communication Thread main loop */
/*-------------------------------------------------------------------*/
static void *commadpt_thread(void *vca)
{
COMMADPT *ca; /* Work CA Control Block Pointer */
int devnum; /* device number copy for convenience*/
int rc; /* return code from various rtns */
char threadname[40]; /* string for WRMSG */
/*---------------------END OF DECLARES---------------------------*/
/* fetch the commadpt structure */
ca=(COMMADPT *)vca;
/* Obtain the CA lock */
obtain_lock(&ca->lock);
/* get a work copy of devnum (for messages) */
devnum=ca->devnum;
MSGBUF(threadname, "3705 device(%1d:%04X) thread", ca->dev->ssid, devnum);
WRMSG(HHC00100, "I", thread_id(), get_thread_priority(0), threadname);
while (!sysblk.shutdown) {
release_lock(&ca->lock);
usleep(50000 + (ca->unack_attn_count * 100000));
obtain_lock(&ca->lock);
make_sna_requests2(ca);
make_sna_requests3(ca);
if (ca->sendq
// attempt to fix hot i/o bug
&& ca->unack_attn_count < 6
)
{
ca->unack_attn_count++;
rc = device_attention(ca->dev, CSW_ATTN);
if(ca->dev->ccwtrace)
WRMSG(HHC01057, "D",
SSID_TO_LCSS(ca->dev->ssid), ca->dev->devnum, rc);
}
}
// end while(!sysblk.shutdown)
WRMSG(HHC00101, "I", thread_id(), get_thread_priority(0), threadname);
release_lock(&ca->lock);
return NULL;
}
/*-------------------------------------------------------------------*/
/* Halt currently executing I/O command */
/*-------------------------------------------------------------------*/
static void commadpt_halt(DEVBLK *dev)
{
if(!dev->busy)
{
return;
}
}
/* The following 3 MSG functions ensure only 1 (one) */
/* hardcoded instance exist for the same numbered msg */
/* that is issued on multiple situations */
static void msg013e(DEVBLK *dev,char *kw,char *kv)
{
WRMSG(HHC01007, "E", SSID_TO_LCSS(dev->ssid), dev->devnum,kw,kv);
}
/*-------------------------------------------------------------------*/
/* Device Initialisation */
/*-------------------------------------------------------------------*/
static int commadpt_init_handler (DEVBLK *dev, int argc, char *argv[])
{
char thread_name[32];
char thread_name2[32];
int i;
int rc;
int pc; /* Parse code */
int errcnt;
union {
int num;
char text[80];
} res;
/* For re-initialisation, close the existing file, if any */
if (dev->fd >= 0)
(dev->hnd->close)(dev);
dev->devtype=0x3705;
dev->excps = 0;
if(dev->ccwtrace)
{
WRMSG(HHC01058,"D",
SSID_TO_LCSS(dev->ssid), dev->devnum);
}
if(dev->commadpt!=NULL)
{
commadpt_clean_device(dev);
}
rc=commadpt_alloc_device(dev);
if(rc<0)
{
WRMSG(HHC01011, "I", SSID_TO_LCSS(dev->ssid), dev->devnum);
return(-1);
}
if(dev->ccwtrace)
{
WRMSG(HHC01059,"D",
SSID_TO_LCSS(dev->ssid), dev->devnum);
}
errcnt=0;
/*
* Initialise ports & hosts
*/
dev->commadpt->sfd=-1;
dev->commadpt->lport=0;
dev->commadpt->debug_sna=0;
dev->commadpt->emu3791=0;
for(i=0;i<argc;i++)
{
pc=parser(ptab,argv[i],&res);
if(pc<0)
{
WRMSG(HHC01012, "E",SSID_TO_LCSS(dev->ssid), dev->devnum,argv[i]);
errcnt++;
continue;
}
if(pc==0)
{
WRMSG(HHC01019, "E",SSID_TO_LCSS(dev->ssid), dev->devnum,argv[i]);
errcnt++;
continue;
}
switch(pc)
{
case COMMADPT_KW_DEBUG:
if (res.text[0] == 'y' || res.text[0] == 'Y')
dev->commadpt->debug_sna = 1;
else
dev->commadpt->debug_sna = 0;
break;
case COMMADPT_KW_LPORT:
rc=commadpt_getport(res.text);
if(rc<0)
{
errcnt++;
msg013e(dev,"LPORT",res.text);
break;
}
dev->commadpt->lport=rc;
break;
case COMMADPT_KW_LHOST:
if(strcmp(res.text,"*")==0)
{
dev->commadpt->lhost=INADDR_ANY;
break;
}
rc=commadpt_getaddr(&dev->commadpt->lhost,res.text);
if(rc!=0)
{
msg013e(dev,"LHOST",res.text);
errcnt++;
}
break;
case COMMADPT_KW_EMU3791:
if(strcasecmp(res.text,"yes")==0 || strcmp(res.text,"1"))
dev->commadpt->emu3791=1;
break;
default:
break;
}
}
if(errcnt>0)
{
WRMSG(HHC01014, "I",SSID_TO_LCSS(dev->ssid),dev->devnum);
return -1;
}
dev->bufsize=256;
dev->numsense=2;
memset(dev->sense,0,sizeof(dev->sense));
init_bufpool(dev->commadpt);
dev->commadpt->devnum=dev->devnum;
/* Initialize the CA lock */
initialize_lock(&dev->commadpt->lock);
/* Initialise thread->I/O & halt initiation EVB */
initialize_condition(&dev->commadpt->ipc);
initialize_condition(&dev->commadpt->ipc_halt);
/* Allocate I/O -> Thread signaling pipe */
create_pipe(dev->commadpt->pipe);
/* Obtain the CA lock */
obtain_lock(&dev->commadpt->lock);
/* Start the telnet worker thread */
/* Set thread-name for debugging purposes */
snprintf(thread_name2,sizeof(thread_name2),
"commadpt %1d:%04X thread2",dev->ssid,dev->devnum);
thread_name2[sizeof(thread_name2)-1]=0;
rc = create_thread(&dev->commadpt->tthread,&sysblk.detattr,telnet_thread,dev->commadpt,thread_name2);
if(rc)
{
WRMSG(HHC00102, "E" ,strerror(rc));
release_lock(&dev->commadpt->lock);
return -1;
}
/* Start the async worker thread */
/* Set thread-name for debugging purposes */
snprintf(thread_name,sizeof(thread_name),
"commadpt %1d:%04X thread",dev->ssid,dev->devnum);
thread_name[sizeof(thread_name)-1]=0;
rc = create_thread(&dev->commadpt->cthread,&sysblk.detattr,commadpt_thread,dev->commadpt,thread_name);
if(rc)
{
WRMSG(HHC00102, "E", strerror(rc));
release_lock(&dev->commadpt->lock);
return -1;
}
dev->commadpt->have_cthread=1;
/* Release the CA lock */
release_lock(&dev->commadpt->lock);
/* Indicate succesfull completion */
return 0;
}
/*-------------------------------------------------------------------*/
/* Query the device definition */
/*-------------------------------------------------------------------*/
static void commadpt_query_device (DEVBLK *dev, char **class,
int buflen, char *buffer)
{
*class = "LINE";
snprintf(buffer,buflen-1,"Read count=%d, Write count=%d IO[%" I64_FMT "u]",
dev->commadpt->read_ccw_count, dev->commadpt->write_ccw_count, dev->excps );
}
/*-------------------------------------------------------------------*/
/* Close the device */
/* Invoked by HERCULES shutdown & DEVINIT processing */
/*-------------------------------------------------------------------*/
static int commadpt_close_device ( DEVBLK *dev )
{
if(dev->ccwtrace)
{
WRMSG(HHC01060,"D",SSID_TO_LCSS(dev->ssid), dev->devnum);
}
/* Obtain the CA lock */
obtain_lock(&dev->commadpt->lock);
/* Terminate current I/O thread if necessary */
if(dev->busy)
{
commadpt_halt(dev);
}
free_bufpool(dev->commadpt);
/* release the CA lock */
release_lock(&dev->commadpt->lock);
/* Free all work storage */
commadpt_clean_device(dev);
/* Indicate to hercules the device is no longer opened */
dev->fd=-1;
if(dev->ccwtrace)
{
WRMSG(HHC01061,"D",SSID_TO_LCSS(dev->ssid), dev->devnum);
}
return 0;
}
void make_seq (COMMADPT * ca, BYTE * reqptr) {
if (reqptr[4] == 0x38) {
reqptr[6] = (unsigned char)(++ca->ncpa_sscp_seqn >> 8) & 0xff;
reqptr[7] = (unsigned char)( ca->ncpa_sscp_seqn ) & 0xff;
} else {
reqptr[6] = (unsigned char)(++ca->ncpb_sscp_seqn >> 8) & 0xff;
reqptr[7] = (unsigned char)( ca->ncpb_sscp_seqn ) & 0xff;
}
}
static void format_sna (BYTE * requestp, char * tag, U16 ssid, U16 devnum) {
char fmtbuf[32];
char fmtbuf2[32];
char fmtbuf3[32];
char fmtbuf4[32];
// char fmtbuf5[256];
char fmtbuf6[32];
char *ru_type="";
int len;
sprintf(fmtbuf, "%02X%02X %02X%02X %02X%02X %02X%02X %02X%02X",
requestp[0], requestp[1], requestp[2], requestp[3], requestp[4], requestp[5], requestp[6], requestp[7], requestp[8], requestp[9]);
sprintf(fmtbuf2, "%02X%02X%02X",
requestp[10], requestp[11], requestp[12]);
len = (requestp[8] << 8) + requestp[9];
len -= 3; /* for len of ru only */
sprintf(fmtbuf3, "%02X", requestp[13]);
sprintf(fmtbuf4, "%02X", requestp[14]);
if (len > 1)
strlcat(fmtbuf3, fmtbuf4, sizeof(fmtbuf3));
sprintf(fmtbuf4, "%02X", requestp[15]);
if (len > 2)
strlcat(fmtbuf3, fmtbuf4, sizeof(fmtbuf3));
if (requestp[13] == 0x11)
ru_type = "ACTPU";
if (requestp[13] == 0x0D)
ru_type = "ACTLU";
if (requestp[13] == 0x0E)
ru_type = "DACTLU";
if (requestp[13] == 0x12)
ru_type = "DACTPU";
if (requestp[13] == 0xA0)
ru_type = "SDT";
if (requestp[13] == 0x31)
ru_type = "BIND";
if (requestp[13] == 0x32)
ru_type = "UNBIND";
if (!memcmp(&requestp[13], R010201, 3))
ru_type = "CONTACT";
if (!memcmp(&requestp[13], R010202, 3))
ru_type = "DISCONTACT";
if (!memcmp(&requestp[13], R010203, 3))
ru_type = "IPLINIT";
if (!memcmp(&requestp[13], R010204, 3))
ru_type = "IPLTEXT";
if (!memcmp(&requestp[13], R010205, 3))
ru_type = "IPLFINAL";
if (!memcmp(&requestp[13], R01020A, 3))
ru_type = "ACTLINK";
if (!memcmp(&requestp[13], R01020B, 3))
ru_type = "DACTLINK";
if (!memcmp(&requestp[13], R010211, 3)) {
sprintf(fmtbuf6, "%s[%02x]", "SETCV", requestp[18]);
ru_type = fmtbuf6;
if ((requestp[10] & 0x80) != 0)
ru_type = "SETCV";
}
if (!memcmp(&requestp[13], R010280, 3))
ru_type = "CONTACTED";
if (!memcmp(&requestp[13], R010281, 3))
ru_type = "INOP";
if (!memcmp(&requestp[13], R010284, 3))
ru_type = "REQCONT";
if (!memcmp(&requestp[13], R01021B, 3))
ru_type = "REQDISCONT";
if (!memcmp(&requestp[13], R01021A, 3))
ru_type = "FNA";
if (!memcmp(&requestp[13], R01020F, 3))
ru_type = "ABCONN";
if (!memcmp(&requestp[13], R010219, 3))
ru_type = "ANA";
if (!memcmp(&requestp[13], R010216, 3))
ru_type = "ACTCONNIN";
if (!memcmp(&requestp[13], R010217, 3))
ru_type = "DACTCONNIN";
if ((requestp[10] & 0x08) == 0)
ru_type = "";
WRMSG(HHC01062,"D",
SSID_TO_LCSS(ssid), devnum, tag, fmtbuf, fmtbuf2, fmtbuf3, ru_type);
}
static void make_sna_requests2 (COMMADPT *ca) {
BYTE *respbuf;
BYTE *ru_ptr;
int ru_size;
void *eleptr;
int bufp = 0;
while (ca->inpbufl > 0) {
eleptr = get_bufpool(&ca->freeq);
if (!eleptr) {
WRMSG(HHC01020, "E", SSID_TO_LCSS(ca->dev->ssid), ca->dev->devnum, "SNA request2");
return;
}
respbuf = SIZEOF_INT_P + (BYTE*)eleptr;
/* first do the ten-byte FID1 TH */
respbuf[0] = 0x1C;
respbuf[1] = 0x00;
respbuf[2] = ca->tso_addr0; // daf
respbuf[3] = ca->tso_addr1;
respbuf[4] = ca->lu_addr0; // oaf
respbuf[5] = ca->lu_addr1; // oaf
respbuf[6] = (unsigned char)(++ca->lu_lu_seqn >> 8) & 0xff;
respbuf[7] = (unsigned char)( ca->lu_lu_seqn ) & 0xff;
/* do RH */
respbuf[10] = 0x00;
if (!bufp) {
respbuf[10] |= 0x02; /* set first in chain */
}
respbuf[11] = 0x90;
respbuf[12] = 0x00;
/* do RU */
// FIXME - max. ru_size should be based on BIND settings
// A true fix would also require code changes to READ CCW processing
// including possibly (gasp) segmenting long PIUs into multiple BTUs
ru_size = min(256-(BUFPD+10+3),ca->inpbufl);
ru_ptr = &respbuf[13];
if (!ca->bindflag) {
// send as character-coded logon to SSCP
if (ru_size > 0 && (ca->inpbuf[ca->inpbufl-1] == 0x0d || ca->inpbuf[ca->inpbufl-1] == 0x25)) {
ru_size--;
}
if (ru_size > 0 && (ca->inpbuf[ca->inpbufl-1] == 0x0d || ca->inpbuf[ca->inpbufl-1] == 0x25)) {
ru_size--;
}
respbuf[2] = ca->sscp_addr0;
respbuf[3] = ca->sscp_addr1;
respbuf[11] = 0x80;
respbuf[12] = 0x00;
}
memcpy(ru_ptr, &ca->inpbuf[bufp], ru_size);
bufp += ru_size;
ca->inpbufl -= ru_size;
if (!ca->is_3270) {
ca->inpbufl = 0;
}
if (!ca->inpbufl) {
respbuf[10] |= 0x01; /* set last in chain */
if (ca->bindflag) {
respbuf[12] |= 0x20; /* set CD */
}
}
/* set length field in TH */
ru_size += 3; /* for RH */
respbuf[8] = (unsigned char)(ru_size >> 8) & 0xff;
respbuf[9] = (unsigned char)(ru_size ) & 0xff;
put_bufpool(&ca->sendq, eleptr);
} /* end of while (ca->inpbufl > 0) */
}
static void make_sna_requests3 (COMMADPT *ca) {
BYTE *respbuf;
BYTE *ru_ptr;
int ru_size;
void *eleptr;
if (!ca->telnet_int) return;
eleptr = get_bufpool(&ca->freeq);
if (!eleptr) {
WRMSG(HHC01020, "E", SSID_TO_LCSS(ca->dev->ssid), ca->dev->devnum, "SNA request3");
return;
}
respbuf = SIZEOF_INT_P + (BYTE*)eleptr;
/* first do the ten-byte FID1 TH */
respbuf[0] = 0x1D;
respbuf[1] = 0x00;
respbuf[2] = ca->tso_addr0; // daf
respbuf[3] = ca->tso_addr1;
respbuf[4] = ca->lu_addr0; // oaf
respbuf[5] = ca->lu_addr1; // oaf
respbuf[6] = 0x11;
respbuf[7] = 0x11;
/* do RH */
respbuf[10] = 0x4B;
respbuf[11] = 0x80;
respbuf[12] = 0x00;
/* do RU */
ru_size = 0;
ru_ptr = &respbuf[13];
ru_ptr[ru_size++] = 0xc9; // SIG
ru_ptr[ru_size++] = 0x00;
ru_ptr[ru_size++] = 0x01;
ru_size += 3; /* for RH */
respbuf[8] = (unsigned char)(ru_size >> 8) & 0xff;
respbuf[9] = (unsigned char)(ru_size ) & 0xff;
put_bufpool(&ca->sendq, eleptr);
ca->telnet_int = 0;
}
static void make_sna_requests4 (COMMADPT *ca, int flag, BYTE pu_type) {
/* send type flag: 0=REQCONT 1=REQDISCONT */
BYTE *respbuf;
BYTE *ru_ptr;
int ru_size;
void *eleptr;
eleptr = get_bufpool(&ca->freeq);
if (!eleptr) {
WRMSG(HHC01020, "E", SSID_TO_LCSS(ca->dev->ssid), ca->dev->devnum, "SNA request4");
return;
}
respbuf = SIZEOF_INT_P + (BYTE*)eleptr;
/* first do the ten-byte FID1 TH */
respbuf[0] = 0x1C;
respbuf[1] = 0x00;
respbuf[2] = ca->sscp_addr0; // daf
respbuf[3] = ca->sscp_addr1;
// set oaf
if (flag == 0) {
respbuf[4] = ca->ncp_addr0;
respbuf[5] = ca->ncp_addr1;
make_seq(ca, respbuf);
} else {
respbuf[4] = ca->pu_addr0;
respbuf[5] = ca->pu_addr1;
respbuf[6] = 0x00;
respbuf[7] = 0x01;
}
/* do RH */
respbuf[10] = 0x0b;
respbuf[11] = 0x00;
respbuf[12] = 0x00;
/* do RU */
ru_size = 0;
ru_ptr = &respbuf[13];
if (flag == 0) {
ru_ptr[ru_size++] = 0x01; // REQCONT (REQUEST CONTACT)
ru_ptr[ru_size++] = 0x02;
ru_ptr[ru_size++] = 0x84;
ru_ptr[ru_size++] = 0x40; // network address of link
ru_ptr[ru_size++] = 0x01;
ru_ptr[ru_size++] = pu_type; // PU type
ru_ptr[ru_size++] = 0x00;
ru_ptr[ru_size++] = 0x01; // IDBLK=017,IDNUM=00017
ru_ptr[ru_size++] = 0x70;
ru_ptr[ru_size++] = 0x00;
ru_ptr[ru_size++] = 0x17;
} else {
ru_ptr[ru_size++] = 0x01; // REQDISCONT (REQUEST DISCONTACT)
ru_ptr[ru_size++] = 0x02;
ru_ptr[ru_size++] = 0x1B;
ru_ptr[ru_size++] = 0x00;
}
ru_size += 3; /* for RH */
respbuf[8] = (unsigned char)(ru_size >> 8) & 0xff;
respbuf[9] = (unsigned char)(ru_size ) & 0xff;
put_bufpool(&ca->sendq, eleptr);
ca->telnet_int = 0;
}
static void make_sna_requests5 (COMMADPT *ca) {
BYTE *respbuf;
BYTE *ru_ptr;
int ru_size;
void *eleptr;
eleptr = get_bufpool(&ca->freeq);
if (!eleptr) {
WRMSG(HHC01020, "E", SSID_TO_LCSS(ca->dev->ssid), ca->dev->devnum, "SNA request5");
return;
}
respbuf = SIZEOF_INT_P + (BYTE*)eleptr;
/* first do the ten-byte FID1 TH */
respbuf[0] = 0x1C;
respbuf[1] = 0x00;
respbuf[2] = ca->sscp_addr0; // daf
respbuf[3] = ca->sscp_addr1;
respbuf[4] = ca->ncp_addr0; // oaf
respbuf[5] = ca->ncp_addr1;
// set seq no.
make_seq(ca, respbuf);
/* do RH */
respbuf[10] = 0x0B;
respbuf[11] = 0x00;
respbuf[12] = 0x00;
/* do RU */
ru_size = 0;
ru_ptr = &respbuf[13];
ru_ptr[ru_size++] = 0x01; // INOP
ru_ptr[ru_size++] = 0x02;
ru_ptr[ru_size++] = 0x81;
ru_ptr[ru_size++] = ca->pu_addr0;
ru_ptr[ru_size++] = ca->pu_addr1;
ru_ptr[ru_size++] = 0x01; // format/reason
ru_size += 3; /* for RH */
respbuf[8] = (unsigned char)(ru_size >> 8) & 0xff;
respbuf[9] = (unsigned char)(ru_size ) & 0xff;
put_bufpool(&ca->sendq, eleptr);
}
void make_sna_requests (BYTE * requestp, COMMADPT *ca) {
BYTE *respbuf;
BYTE *ru_ptr;
int ru_size;
void *eleptr;
if (memcmp(&requestp[13], R010201, 3)) return; // we only want to process CONTACT
eleptr = get_bufpool(&ca->freeq);
if (!eleptr) {
WRMSG(HHC01020, "E", SSID_TO_LCSS(ca->dev->ssid), ca->dev->devnum, "SNA request");
return;
}
respbuf = SIZEOF_INT_P + (BYTE*)eleptr;
/* first do the ten-byte FID1 TH */
// respbuf[0] = requestp[0];
// respbuf[1] = requestp[1];
respbuf[0] = 0x1c;
respbuf[1] = 0x00;
respbuf[2] = requestp[4]; // daf
respbuf[3] = requestp[5];
respbuf[4] = requestp[2]; // oaf
respbuf[5] = requestp[3];
make_seq(ca, respbuf);
/* do RH */
respbuf[10] = requestp[10];
respbuf[11] = requestp[11];
respbuf[11] = 0x00;
respbuf[12] = requestp[12];
/* make a CONTACTED RU */
ru_size = 0;
ru_ptr = &respbuf[13];
ru_ptr[ru_size++] = 0x01;
ru_ptr[ru_size++] = 0x02;
ru_ptr[ru_size++] = 0x80;
ru_ptr[ru_size++] = requestp[16];
ru_ptr[ru_size++] = requestp[17];
ru_ptr[ru_size++] = 0x01;
/* set length field in TH */
ru_size += 3; /* for RH */
respbuf[8] = (unsigned char)(ru_size >> 8) & 0xff;
respbuf[9] = (unsigned char)(ru_size ) & 0xff;
put_bufpool(&ca->sendq, eleptr);
}
void make_sna_response (BYTE * requestp, COMMADPT *ca) {
BYTE *respbuf;
BYTE *ru_ptr;
int ru_size;
void *eleptr;
BYTE obuf[4096];
BYTE buf[BUFLEN_3270];
int amt;
int i1;
if ((requestp[10] & 0x80) != 0) return; // disregard if this is a resp.
if ((requestp[10] & (unsigned char)0xfc) == 0x00 && requestp[2] == ca->lu_addr0 && requestp[3] == ca->lu_addr1 && ca->sfd > 0) { /* if type=data, and DAF matches up, and socket exists */
amt = (requestp[8] << 8) + requestp[9];
amt -= 3;
if (ca->is_3270) {
memcpy(buf, &requestp[13], amt);
/* Double up any IAC bytes in the data */
amt = double_up_iac (buf, amt);
/* Append telnet EOR marker at end of data */
if ((requestp[10] & 0x01) == 0x01) { /* if last-in-chain is set */
buf[amt++] = IAC;
buf[amt++] = EOR_MARK;
}
/* Send the data to the client */
logdump ("SEND", ca->dev, buf, amt);
write_socket(ca->sfd,buf,amt);
} else {
// convert data portion to ASCII and write to remote user
if (amt > 0) {
memcpy(obuf, &requestp[13], amt);
for (i1=0; i1<amt; i1++) {
obuf[i1] = guest_to_host(obuf[i1]);
}
logdump ("SEND", ca->dev, obuf, amt);
write_socket(ca->sfd,obuf,amt);
}
}
}
if ((requestp[11] & 0xf0) != 0x80) return; // disregard if not DR1 requested
eleptr = get_bufpool(&ca->freeq);
if (!eleptr) {
WRMSG(HHC01020, "E", SSID_TO_LCSS(ca->dev->ssid), ca->dev->devnum, "SNA response");
return;
}
respbuf = SIZEOF_INT_P + (BYTE*)eleptr;
/* first do the ten-byte FID1 TH */
respbuf[0] = requestp[0];
respbuf[1] = requestp[1];
respbuf[2] = requestp[4]; // daf
respbuf[3] = requestp[5];
respbuf[4] = requestp[2]; // oaf
respbuf[5] = requestp[3];
respbuf[6] = requestp[6]; // seq #
respbuf[7] = requestp[7];
/* do RH */
respbuf[10] = requestp[10];
respbuf[10] |= 0x83; // indicate this is a resp.
respbuf[11] = requestp[11];
// respbuf[12] = requestp[12];
respbuf[12] = 0x00;
/* do RU */
ru_size = 0;
ru_ptr = &respbuf[13];
if ((requestp[10] & 0x08) != 0)
ru_ptr[ru_size++] = requestp[13];
if (requestp[13] == 0x11 && requestp[14] == 0x02) { /* ACTPU (NCP)*/
ca->ncp_addr0 = requestp[2];
ca->ncp_addr1 = requestp[3];
// ca->ncp_sscp_seqn = 0;
ru_ptr[ru_size++] = 0x02;
if (requestp[2] == 0x40) { /* DAF subarea field = 8? */
ru_ptr[ru_size++] = 0xd4; /* DS CL8'MHPRMT1 ' - NCP load mod name */
ru_ptr[ru_size++] = 0xc8;
ru_ptr[ru_size++] = 0xd7;
ru_ptr[ru_size++] = 0xd9;
ru_ptr[ru_size++] = 0xd4;
ru_ptr[ru_size++] = 0xe3;
ru_ptr[ru_size++] = 0xf1;
ru_ptr[ru_size++] = 0x40;
ca->ncpb_sscp_seqn = 0;
} else {
ru_ptr[ru_size++] = 0xd4; /* DS CL8'MHP3705 ' - NCP load mod name */
ru_ptr[ru_size++] = 0xc8; /* checked by host at ACTPU time */
ru_ptr[ru_size++] = 0xd7;
ru_ptr[ru_size++] = 0xf3;
ru_ptr[ru_size++] = 0xf7;
ru_ptr[ru_size++] = 0xf0;
ru_ptr[ru_size++] = 0xf5;
ru_ptr[ru_size++] = 0x40;
ca->ncpa_sscp_seqn = 0;
}
}
if (requestp[13] == 0x11 && requestp[14] == 0x01) { /* ACTPU (PU)*/
ru_ptr[ru_size++] = 0x01;
/* save daf as our own net addr */
ca->pu_addr0 = requestp[2];
ca->pu_addr1 = requestp[3];
}
if (requestp[13] == 0x01) { /* 01XXXX Network Services */
ru_ptr[ru_size++] = requestp[14];
ru_ptr[ru_size++] = requestp[15];
}
if (!memcmp(&requestp[13], R010219, 3) && ca->sfd > 0) { /* ANA */
if (!ca->is_3270) {
connect_message(ca->sfd, (requestp[20] << 8) + requestp[21], 0);
}
}
if (requestp[13] == 0x0D) { /* ACTLU */
/* save daf as our own net addr */
ca->lu_addr0 = requestp[2];
ca->lu_addr1 = requestp[3];
/* save oaf as our sscp net addr */
ca->sscp_addr0 = requestp[4];
ca->sscp_addr1 = requestp[5];
ca->lu_sscp_seqn = 0;
ca->bindflag = 0;
}
if (requestp[13] == 0x0E || !memcmp(&requestp[13], R01020F, 3)) { // DACTLU or ABCONN
if (!ca->is_3270) {
connect_message(ca->sfd, 0, 1);
}
ca->hangup = 1;
}
if (requestp[13] == 0x31) { /* BIND */
/* save oaf from BIND request */
ca->tso_addr0 = requestp[4];
ca->tso_addr1 = requestp[5];
ca->lu_lu_seqn = 0;
ca->bindflag = 1;
}
if (requestp[13] == 0x32 && requestp[14] != 0x02) { /* BIND */
ca->bindflag = 0;
}
#if 0
if (requestp[13] == 0x32 && requestp[14] == 0x01 && ca->sfd > 0) { /* UNBIND */
close_socket(ca->sfd);
ca->sfd=-1;
}
#endif
/* set length field in TH */
ru_size += 3; /* for RH */
respbuf[8] = (unsigned char)(ru_size >> 8) & 0xff;
respbuf[9] = (unsigned char)(ru_size ) & 0xff;
put_bufpool(&ca->sendq, eleptr);
}
enum fid_remap {
MAP_FID1_FID2,
MAP_FID2_FID1
};
static void th_remap(enum fid_remap r, BYTE * thptr)
{ /* for 3791 support, remaps SNA FID1 <--> FID2 TH headers */
int thmpf;
int thm2;
int thdaf;
int thoaf;
int thsnf;
int len;
if (r == MAP_FID1_FID2)
{
thmpf = thptr[0];
thm2 = thptr[1];
thdaf = (thptr[2] << 8) + thptr[3];
thoaf = (thptr[4] << 8) + thptr[5];
thsnf = (thptr[6] << 8) + thptr[7];
len = (thptr[8] << 8) + thptr[9];
len += 10;
thptr[0] = (len >> 8) & 0xff;
thptr[1] = len & 0xff;
thptr[2] = 0x00;
thptr[3] = 0x00;
thptr[4] = 0x20 | (thmpf & 0x0f);
thptr[5] = thm2;
thptr[6] = thdaf & 0xff;
thptr[7] = thoaf & 0xff;
thptr[8] = (thsnf >> 8) & 0xff;
thptr[9] = thsnf & 0xff;
}
else
{ /* map fid2 to fid1 */
len = (thptr[0] << 8) + thptr[1];
thmpf = thptr[4];
thm2 = thptr[5];
thdaf = thptr[6];
thoaf = thptr[7];
thsnf = (thptr[8] << 8) + thptr[9];
thdaf |= 0x3800; /* subarea = 7 (maxsuba=31) */
thoaf |= 0x0800; /* subarea = 1 (maxsuba=31) */
len -= 10;
thptr[0] = 0x10 | (thmpf & 0x0f);
thptr[1] = thm2;
thptr[2] = (thdaf >> 8) & 0xff;
thptr[3] = thdaf & 0xff;
thptr[4] = (thoaf >> 8) & 0xff;
thptr[5] = thoaf & 0xff;
thptr[6] = (thsnf >> 8) & 0xff;
thptr[7] = thsnf & 0xff;
thptr[8] = (len >> 8) & 0xff;
thptr[9] = len & 0xff;
}
}
/*-------------------------------------------------------------------*/
/* Execute a Channel Command Word */
/*-------------------------------------------------------------------*/
static void commadpt_execute_ccw (DEVBLK *dev, BYTE code, BYTE flags,
BYTE chained, U32 count, BYTE prevcode, int ccwseq,
BYTE *iobuf, BYTE *more, BYTE *unitstat, U32 *residual)
{
U32 num; /* Work : Actual CCW transfer count */
BYTE *piudata;
int piusize;
void *eleptr;
int llsize;
UNREFERENCED(flags);
UNREFERENCED(chained);
UNREFERENCED(prevcode);
UNREFERENCED(ccwseq);
*residual = 0;
/*
* Obtain the COMMADPT lock
*/
if(dev->ccwtrace)
{
WRMSG(HHC01063,"D",
SSID_TO_LCSS(dev->ssid), dev->devnum,code);
}
obtain_lock(&dev->commadpt->lock);
switch (code) {
/*---------------------------------------------------------------*/
/* BASIC SENSE */
/*---------------------------------------------------------------*/
case 0x04:
dev->commadpt->unack_attn_count = 0;
num=count<dev->numsense?count:dev->numsense;
*more=count<dev->numsense?1:0;
memcpy(iobuf,dev->sense,num);
*residual=count-num;
*unitstat=CSW_CE|CSW_DE;
break;
/*---------------------------------------------------------------*/
/* READ type CCWs */
/*---------------------------------------------------------------*/
case 0x02: /* READ */
dev->commadpt->read_ccw_count++;
dev->commadpt->unack_attn_count = 0;
*more = 0;
make_sna_requests2(dev->commadpt);
make_sna_requests3(dev->commadpt);
eleptr = get_bufpool(&dev->commadpt->sendq);
*residual=count;
if (eleptr) {
piudata = SIZEOF_INT_P + (BYTE*)eleptr;
piusize = (piudata[8] << 8) + piudata[9];
piusize += 10; // for FID1 TH
iobuf[0] = BUFPD;
memcpy (&iobuf[BUFPD], piudata, piusize);
if (dev->commadpt->emu3791) {
llsize = piusize + BUFPD;
iobuf[0] = (llsize >> 8) & 0xff;
iobuf[1] = llsize & 0xff;
th_remap(MAP_FID1_FID2, &iobuf[BUFPD]);
}
*residual=count - (piusize + BUFPD);
logdump("READ", dev, &iobuf[BUFPD], piusize);
if (dev->commadpt->debug_sna)
format_sna(piudata, "RD", dev->ssid, dev->devnum);
put_bufpool(&dev->commadpt->freeq, eleptr);
}
*unitstat=CSW_CE|CSW_DE;
#if 0
if (dev->commadpt->sendq) {
*unitstat|=CSW_ATTN;
}
#endif
*unitstat|=CSW_UX;
break;
/*---------------------------------------------------------------*/
/* 3791 WRITE BLOCK */
/*---------------------------------------------------------------*/
case 0x05:
logdump("WRITE BLOCK", dev, iobuf, count);
*residual=0;
*unitstat=CSW_CE|CSW_DE;
break;
/*---------------------------------------------------------------*/
/* WRITE type CCWs */
/*---------------------------------------------------------------*/
case 0x09: /* WRITE BREAK */
case 0x01: /* WRITE */
dev->commadpt->write_ccw_count++;
dev->commadpt->unack_attn_count = 0;
logdump("WRITE", dev, iobuf, count);
if (dev->commadpt->emu3791 && (iobuf[4] & 0xf0) == 0x20)
th_remap(MAP_FID2_FID1, iobuf);
if ((iobuf[0] & 0xf0) == 0x10) { // if FID1
if (dev->commadpt->debug_sna)
format_sna(iobuf, "WR", dev->ssid, dev->devnum);
make_sna_response(iobuf, dev->commadpt);
make_sna_requests(iobuf, dev->commadpt);
}
*residual=0;
*unitstat=CSW_CE|CSW_DE;
#if 0
if (dev->commadpt->sendq) {
*unitstat|=CSW_ATTN;
*unitstat|=CSW_UX|CSW_ATTN;
}
#endif
break;
/*---------------------------------------------------------------*/
/* CCWs to be treated as NOPs */
/*---------------------------------------------------------------*/
case 0x03: /* NOP */
case 0x93: /* RESTART */
case 0x31: /* WS0 */
case 0x51: /* WS1 */
case 0x32: /* RS0 */
case 0x52: /* RS1 */
*residual=count;
*unitstat=CSW_CE|CSW_DE;
break;
default:
/*---------------------------------------------------------------*/
/* INVALID OPERATION */
/*---------------------------------------------------------------*/
/* Set command reject sense byte, and unit check status */
*unitstat=CSW_CE+CSW_DE+CSW_UC;
dev->sense[0]=SENSE_CR;
break;
}
release_lock(&dev->commadpt->lock);
}
/*---------------------------------------------------------------*/
/* DEVICE FUNCTION POINTERS */
/*---------------------------------------------------------------*/
#if defined(OPTION_DYNAMIC_LOAD)
static
#endif
DEVHND com3705_device_hndinfo = {
&commadpt_init_handler, /* Device Initialisation */
&commadpt_execute_ccw, /* Device CCW execute */
&commadpt_close_device, /* Device Close */
&commadpt_query_device, /* Device Query */
NULL, /* Device Extended Query */
NULL, /* Device Start channel pgm */
NULL, /* Device End channel pgm */
NULL, /* Device Resume channel pgm */
NULL, /* Device Suspend channel pgm */
&commadpt_halt, /* Device Halt channel pgm */
NULL, /* Device Read */
NULL, /* Device Write */
NULL, /* Device Query used */
NULL, /* Device Reserve */
NULL, /* Device Release */
NULL, /* Device Attention */
commadpt_immed_command, /* Immediate CCW Codes */
NULL, /* Signal Adapter Input */
NULL, /* Signal Adapter Output */
NULL, /* Signal Adapter Sync */
NULL, /* Signal Adapter Output Mult */
NULL, /* QDIO subsys desc */
NULL, /* QDIO set subchan ind */
NULL, /* Hercules suspend */
NULL /* Hercules resume */
};
/* Libtool static name colision resolution */
/* note : lt_dlopen will look for symbol & modulename_LTX_symbol */
#if !defined(HDL_BUILD_SHARED) && defined(HDL_USE_LIBTOOL)
#define hdl_ddev hdt3705_LTX_hdl_ddev
#define hdl_depc hdt3705_LTX_hdl_depc
#define hdl_reso hdt3705_LTX_hdl_reso
#define hdl_init hdt3705_LTX_hdl_init
#define hdl_fini hdt3705_LTX_hdl_fini
#endif /* !defined(HDL_BUILD_SHARED)... */
#if defined(OPTION_DYNAMIC_LOAD)
HDL_DEPENDENCY_SECTION;
{
HDL_DEPENDENCY(HERCULES);
HDL_DEPENDENCY(DEVBLK);
HDL_DEPENDENCY(SYSBLK);
}
END_DEPENDENCY_SECTION;
#if defined(WIN32) && !defined(HDL_USE_LIBTOOL) && !defined(_MSVC_)
#undef sysblk
HDL_RESOLVER_SECTION;
{
HDL_RESOLVE_PTRVAR( psysblk, sysblk );
}
END_RESOLVER_SECTION;
#endif /* defined(WIN32)... */
HDL_DEVICE_SECTION;
{
HDL_DEVICE(3705, com3705_device_hndinfo );
}
END_DEVICE_SECTION;
#endif /* defined(OPTION_DYNAMIC_LOAD) */