1999-02-19 03:21:36 +03:00
/*
* Copyright ( c ) 1993 , 1994 , 1995 , 1996 Rick Sladkey < jrs @ world . std . com >
1999-12-23 17:20:14 +03:00
* Copyright ( c ) 1996 - 1999 Wichert Akkerman < wichert @ cistron . nl >
1999-02-19 03:21:36 +03:00
* All rights reserved .
*
* Redistribution and use in source and binary forms , with or without
* modification , are permitted provided that the following conditions
* are met :
* 1. Redistributions of source code must retain the above copyright
* notice , this list of conditions and the following disclaimer .
* 2. Redistributions in binary form must reproduce the above copyright
* notice , this list of conditions and the following disclaimer in the
* documentation and / or other materials provided with the distribution .
* 3. The name of the author may not be used to endorse or promote products
* derived from this software without specific prior written permission .
*
* THIS SOFTWARE IS PROVIDED BY THE AUTHOR ` ` AS IS ' ' AND ANY EXPRESS OR
* IMPLIED WARRANTIES , INCLUDING , BUT NOT LIMITED TO , THE IMPLIED WARRANTIES
* OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED .
* IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR ANY DIRECT , INDIRECT ,
* INCIDENTAL , SPECIAL , EXEMPLARY , OR CONSEQUENTIAL DAMAGES ( INCLUDING , BUT
* NOT LIMITED TO , PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES ; LOSS OF USE ,
* DATA , OR PROFITS ; OR BUSINESS INTERRUPTION ) HOWEVER CAUSED AND ON ANY
* THEORY OF LIABILITY , WHETHER IN CONTRACT , STRICT LIABILITY , OR TORT
* ( INCLUDING NEGLIGENCE OR OTHERWISE ) ARISING IN ANY WAY OUT OF THE USE OF
* THIS SOFTWARE , EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE .
*
* $ Id $
*/
# include "defs.h"
2001-04-10 14:32:26 +04:00
# ifdef HAVE_POLL_H
# include <poll.h>
# endif
2000-02-01 19:12:33 +03:00
# ifdef HAVE_SYS_POLL_H
1999-02-19 03:21:36 +03:00
# include <sys/poll.h>
2000-02-01 19:12:33 +03:00
# endif
2001-04-10 14:32:26 +04:00
# ifdef HAVE_STROPTS_H
# include <stropts.h>
# endif
# ifdef HAVE_SYS_CONF_H
# include <sys/conf.h>
# endif
# ifdef HAVE_SYS_STREAM_H
# include <sys/stream.h>
# endif
# ifdef HAVE_SYS_TIHDR_H
# include <sys/tihdr.h>
# endif
1999-02-19 03:21:36 +03:00
2001-04-10 14:32:26 +04:00
# if defined(HAVE_SYS_STREAM_H) || defined(linux) || defined(FREEBSD)
# ifndef HAVE_STROPTS_H
1999-02-19 03:21:36 +03:00
# define RS_HIPRI 1
struct strbuf {
int maxlen ; /* no. of bytes in buffer */
int len ; /* no. of bytes returned */
char * buf ; /* pointer to data */
} ;
# define MORECTL 1
# define MOREDATA 2
2001-04-10 14:32:26 +04:00
# endif /* !HAVE_STROPTS_H */
1999-02-19 03:21:36 +03:00
# ifdef HAVE_SYS_TIUSER_H
# include <sys/tiuser.h>
# include <sys/sockmod.h>
# include <sys/timod.h>
# endif /* HAVE_SYS_TIUSER_H */
2000-09-02 01:03:06 +04:00
# ifndef FREEBSD
1999-02-19 03:21:36 +03:00
static struct xlat msgflags [ ] = {
{ RS_HIPRI , " RS_HIPRI " } ,
{ 0 , NULL } ,
} ;
static void
printstrbuf ( tcp , sbp , getting )
struct tcb * tcp ;
struct strbuf * sbp ;
int getting ;
{
if ( sbp - > maxlen = = - 1 & & getting )
tprintf ( " {maxlen=-1} " ) ;
else {
tprintf ( " { " ) ;
if ( getting )
tprintf ( " maxlen=%d, " , sbp - > maxlen ) ;
tprintf ( " len=%d, buf= " , sbp - > len ) ;
1999-05-09 04:29:58 +04:00
printstr ( tcp , ( unsigned long ) sbp - > buf , sbp - > len ) ;
1999-02-19 03:21:36 +03:00
tprintf ( " } " ) ;
}
}
static void
printstrbufarg ( tcp , arg , getting )
struct tcb * tcp ;
int arg ;
int getting ;
{
struct strbuf buf ;
if ( arg = = 0 )
tprintf ( " NULL " ) ;
else if ( umove ( tcp , arg , & buf ) < 0 )
tprintf ( " {...} " ) ;
else
printstrbuf ( tcp , & buf , getting ) ;
tprintf ( " , " ) ;
}
int
sys_putmsg ( tcp )
struct tcb * tcp ;
{
int i ;
if ( entering ( tcp ) ) {
/* fd */
tprintf ( " %ld, " , tcp - > u_arg [ 0 ] ) ;
/* control and data */
for ( i = 1 ; i < 3 ; i + + )
printstrbufarg ( tcp , tcp - > u_arg [ i ] , 0 ) ;
/* flags */
if ( ! printflags ( msgflags , tcp - > u_arg [ 3 ] ) )
tprintf ( " 0 " ) ;
}
return 0 ;
}
int
sys_getmsg ( tcp )
struct tcb * tcp ;
{
int i , flags ;
if ( entering ( tcp ) ) {
/* fd */
tprintf ( " %lu, " , tcp - > u_arg [ 0 ] ) ;
} else {
if ( syserror ( tcp ) ) {
tprintf ( " %#lx, %#lx, %#lx " ,
tcp - > u_arg [ 1 ] , tcp - > u_arg [ 2 ] , tcp - > u_arg [ 3 ] ) ;
return 0 ;
}
/* control and data */
for ( i = 1 ; i < 3 ; i + + )
printstrbufarg ( tcp , tcp - > u_arg [ i ] , 1 ) ;
/* pointer to flags */
if ( tcp - > u_arg [ 3 ] = = 0 )
tprintf ( " NULL " ) ;
else if ( umove ( tcp , tcp - > u_arg [ 3 ] , & flags ) < 0 )
tprintf ( " [?] " ) ;
else {
tprintf ( " [ " ) ;
if ( ! printflags ( msgflags , flags ) )
tprintf ( " 0 " ) ;
tprintf ( " ] " ) ;
}
/* decode return value */
switch ( tcp - > u_rval ) {
case MORECTL :
tcp - > auxstr = " MORECTL " ;
break ;
case MORECTL | MOREDATA :
tcp - > auxstr = " MORECTL|MOREDATA " ;
break ;
case MOREDATA :
tcp - > auxstr = " MORECTL " ;
break ;
default :
tcp - > auxstr = NULL ;
break ;
}
}
return RVAL_HEX | RVAL_STR ;
}
# ifdef HAVE_PUTPMSG
static struct xlat pmsgflags [ ] = {
2000-10-13 16:47:12 +04:00
# ifdef MSG_HIPRI
1999-02-19 03:21:36 +03:00
{ MSG_HIPRI , " MSG_HIPRI " } ,
2000-10-13 16:47:12 +04:00
# endif
# ifdef MSG_AND
1999-02-19 03:21:36 +03:00
{ MSG_ANY , " MSG_ANY " } ,
2000-10-13 16:47:12 +04:00
# endif
# ifdef MSG_BAND
1999-02-19 03:21:36 +03:00
{ MSG_BAND , " MSG_BAND " } ,
2000-10-13 16:47:12 +04:00
# endif
1999-02-19 03:21:36 +03:00
{ 0 , NULL } ,
} ;
int
sys_putpmsg ( tcp )
struct tcb * tcp ;
{
int i ;
if ( entering ( tcp ) ) {
/* fd */
tprintf ( " %ld, " , tcp - > u_arg [ 0 ] ) ;
/* control and data */
for ( i = 1 ; i < 3 ; i + + )
printstrbufarg ( tcp , tcp - > u_arg [ i ] , 0 ) ;
/* band */
tprintf ( " %ld, " , tcp - > u_arg [ 3 ] ) ;
/* flags */
if ( ! printflags ( pmsgflags , tcp - > u_arg [ 4 ] ) )
tprintf ( " 0 " ) ;
}
return 0 ;
}
int
sys_getpmsg ( tcp )
struct tcb * tcp ;
{
int i , flags ;
if ( entering ( tcp ) ) {
/* fd */
tprintf ( " %lu, " , tcp - > u_arg [ 0 ] ) ;
} else {
if ( syserror ( tcp ) ) {
tprintf ( " %#lx, %#lx, %#lx, %#lx " , tcp - > u_arg [ 1 ] ,
tcp - > u_arg [ 2 ] , tcp - > u_arg [ 3 ] , tcp - > u_arg [ 4 ] ) ;
return 0 ;
}
/* control and data */
for ( i = 1 ; i < 3 ; i + + )
printstrbufarg ( tcp , tcp - > u_arg [ i ] , 1 ) ;
/* pointer to band */
printnum ( tcp , tcp - > u_arg [ 3 ] , " %d " ) ;
1999-11-26 12:18:37 +03:00
tprintf ( " , " ) ;
1999-02-19 03:21:36 +03:00
/* pointer to flags */
if ( tcp - > u_arg [ 4 ] = = 0 )
tprintf ( " NULL " ) ;
else if ( umove ( tcp , tcp - > u_arg [ 4 ] , & flags ) < 0 )
tprintf ( " [?] " ) ;
else {
tprintf ( " [ " ) ;
if ( ! printflags ( pmsgflags , flags ) )
tprintf ( " 0 " ) ;
tprintf ( " ] " ) ;
}
/* decode return value */
switch ( tcp - > u_rval ) {
case MORECTL :
tcp - > auxstr = " MORECTL " ;
break ;
case MORECTL | MOREDATA :
tcp - > auxstr = " MORECTL|MOREDATA " ;
break ;
case MOREDATA :
tcp - > auxstr = " MORECTL " ;
break ;
default :
tcp - > auxstr = NULL ;
break ;
}
}
return RVAL_HEX | RVAL_STR ;
}
# endif /* HAVE_PUTPMSG */
2000-09-02 01:03:06 +04:00
# endif /* !FREEBSD */
1999-02-19 03:21:36 +03:00
2000-02-20 02:59:03 +03:00
# ifdef HAVE_SYS_POLL_H
1999-02-19 03:21:36 +03:00
static struct xlat pollflags [ ] = {
2000-02-01 19:12:33 +03:00
# ifdef POLLIN
1999-02-19 03:21:36 +03:00
{ POLLIN , " POLLIN " } ,
{ POLLPRI , " POLLPRI " } ,
{ POLLOUT , " POLLOUT " } ,
# ifdef POLLRDNORM
{ POLLRDNORM , " POLLRDNORM " } ,
# endif
# ifdef POLLWRNORM
{ POLLWRNORM , " POLLWRNORM " } ,
# endif
# ifdef POLLRDBAND
{ POLLRDBAND , " POLLRDBAND " } ,
# endif
# ifdef POLLWRBAND
{ POLLWRBAND , " POLLWRBAND " } ,
# endif
{ POLLERR , " POLLERR " } ,
{ POLLHUP , " POLLHUP " } ,
{ POLLNVAL , " POLLNVAL " } ,
2000-02-01 19:12:33 +03:00
# endif
1999-02-19 03:21:36 +03:00
{ 0 , NULL } ,
} ;
int
sys_poll ( tcp )
struct tcb * tcp ;
{
struct pollfd * pollp ;
if ( exiting ( tcp ) ) {
int i ;
int nfds = tcp - > u_arg [ 1 ] ;
if ( nfds < = 0 ) {
tprintf ( " %#lx, %d, %ld \n " ,
tcp - > u_arg [ 0 ] , nfds , tcp - > u_arg [ 2 ] ) ;
return 0 ;
}
pollp = ( struct pollfd * ) malloc ( nfds * sizeof ( * pollp ) ) ;
if ( pollp = = NULL ) {
fprintf ( stderr , " sys_poll: no memory \n " ) ;
tprintf ( " %#lx, %d, %ld " ,
tcp - > u_arg [ 0 ] , nfds , tcp - > u_arg [ 2 ] ) ;
return 0 ;
}
if ( umoven ( tcp , tcp - > u_arg [ 0 ] ,
( nfds * sizeof ( * pollp ) ) , ( char * ) pollp ) < 0 ) {
tprintf ( " %#lx " , tcp - > u_arg [ 0 ] ) ;
}
else {
tprintf ( " [ " ) ;
for ( i = 0 ; i < nfds ; i + + ) {
if ( i )
tprintf ( " , " ) ;
if ( pollp [ i ] . fd < 0 ) {
tprintf ( " {fd=%d} " , pollp [ i ] . fd ) ;
continue ;
}
tprintf ( " {fd=%d, events= " , pollp [ i ] . fd ) ;
if ( ! printflags ( pollflags , pollp [ i ] . events ) )
tprintf ( " 0 " ) ;
if ( ! syserror ( tcp ) & & pollp [ i ] . revents ) {
tprintf ( " , revents= " ) ;
if ( ! printflags ( pollflags ,
pollp [ i ] . revents ) )
tprintf ( " 0 " ) ;
}
tprintf ( " } " ) ;
}
tprintf ( " ] " ) ;
}
tprintf ( " , %d, " , nfds ) ;
# ifdef INFTIM
if ( tcp - > u_arg [ 2 ] = = INFTIM )
tprintf ( " INFTIM " ) ;
else
# endif
tprintf ( " %ld " , tcp - > u_arg [ 2 ] ) ;
free ( pollp ) ;
}
return 0 ;
}
2000-02-20 02:59:03 +03:00
# else /* !HAVE_SYS_POLL_H */
int
sys_poll ( tcp )
struct tcb * tcp ;
{
return 0 ;
}
# endif
2000-09-02 01:03:06 +04:00
# if !defined(linux) && !defined(FREEBSD)
1999-02-19 03:21:36 +03:00
static struct xlat stream_flush_options [ ] = {
{ FLUSHR , " FLUSHR " } ,
{ FLUSHW , " FLUSHW " } ,
{ FLUSHRW , " FLUSHRW " } ,
# ifdef FLUSHBAND
{ FLUSHBAND , " FLUSHBAND " } ,
# endif
{ 0 , NULL } ,
} ;
static struct xlat stream_setsig_flags [ ] = {
{ S_INPUT , " S_INPUT " } ,
{ S_HIPRI , " S_HIPRI " } ,
{ S_OUTPUT , " S_OUTPUT " } ,
{ S_MSG , " S_MSG " } ,
# ifdef S_ERROR
{ S_ERROR , " S_ERROR " } ,
# endif
# ifdef S_HANGUP
{ S_HANGUP , " S_HANGUP " } ,
# endif
# ifdef S_RDNORM
{ S_RDNORM , " S_RDNORM " } ,
# endif
# ifdef S_WRNORM
{ S_WRNORM , " S_WRNORM " } ,
# endif
# ifdef S_RDBAND
{ S_RDBAND , " S_RDBAND " } ,
# endif
# ifdef S_WRBAND
{ S_WRBAND , " S_WRBAND " } ,
# endif
# ifdef S_BANDURG
{ S_BANDURG , " S_BANDURG " } ,
# endif
{ 0 , NULL } ,
} ;
static struct xlat stream_read_options [ ] = {
{ RNORM , " RNORM " } ,
{ RMSGD , " RMSGD " } ,
{ RMSGN , " RMSGN " } ,
{ 0 , NULL } ,
} ;
static struct xlat stream_read_flags [ ] = {
# ifdef RPROTDAT
{ RPROTDAT , " RPROTDAT " } ,
# endif
# ifdef RPROTDIS
{ RPROTDIS , " RPROTDIS " } ,
# endif
# ifdef RPROTNORM
{ RPROTNORM , " RPROTNORM " } ,
# endif
{ 0 , NULL } ,
} ;
# ifndef RMODEMASK
# define RMODEMASK (~0)
# endif
# ifdef I_SWROPT
static struct xlat stream_write_flags [ ] = {
{ SNDZERO , " SNDZERO " } ,
{ SNDPIPE , " SNDPIPE " } ,
{ 0 , NULL } ,
} ;
# endif /* I_SWROPT */
# ifdef I_ATMARK
static struct xlat stream_atmark_options [ ] = {
{ ANYMARK , " ANYMARK " } ,
{ LASTMARK , " LASTMARK " } ,
{ 0 , NULL } ,
} ;
# endif /* I_ATMARK */
# ifdef TI_BIND
static struct xlat transport_user_options [ ] = {
{ T_CONN_REQ , " T_CONN_REQ " } ,
{ T_CONN_RES , " T_CONN_RES " } ,
{ T_DISCON_REQ , " T_DISCON_REQ " } ,
{ T_DATA_REQ , " T_DATA_REQ " } ,
{ T_EXDATA_REQ , " T_EXDATA_REQ " } ,
{ T_INFO_REQ , " T_INFO_REQ " } ,
{ T_BIND_REQ , " T_BIND_REQ " } ,
{ T_UNBIND_REQ , " T_UNBIND_REQ " } ,
{ T_UNITDATA_REQ , " T_UNITDATA_REQ " } ,
{ T_OPTMGMT_REQ , " T_OPTMGMT_REQ " } ,
{ T_ORDREL_REQ , " T_ORDREL_REQ " } ,
{ 0 , NULL } ,
} ;
2002-05-23 15:48:58 +04:00
static struct xlat transport_user_flags [ ] = {
{ 0 , " 0 " } ,
{ T_MORE , " T_MORE " } ,
{ T_EXPEDITED , " T_EXPEDITED " } ,
{ T_NEGOTIATE , " T_NEGOTIATE " } ,
{ T_CHECK , " T_CHECK " } ,
{ T_DEFAULT , " T_DEFAULT " } ,
{ T_SUCCESS , " T_SUCCESS " } ,
{ T_FAILURE , " T_FAILURE " } ,
{ T_CURRENT , " T_CURRENT " } ,
{ T_PARTSUCCESS , " T_PARTSUCCESS " } ,
{ T_READONLY , " T_READONLY " } ,
{ T_NOTSUPPORT , " T_NOTSUPPORT " } ,
1999-02-19 03:21:36 +03:00
{ 0 , NULL } ,
} ;
2002-05-23 15:48:58 +04:00
# ifdef HAVE_T_OPTHDR
static struct xlat xti_level [ ] = {
{ XTI_GENERIC , " XTI_GENERIC " } ,
{ 0 , NULL } ,
} ;
static struct xlat xti_generic [ ] = {
{ XTI_DEBUG , " XTI_DEBUG " } ,
{ XTI_LINGER , " XTI_LINGER " } ,
{ XTI_RCVBUF , " XTI_RCVBUF " } ,
{ XTI_RCVLOWAT , " XTI_RCVLOWAT " } ,
{ XTI_SNDBUF , " XTI_SNDBUF " } ,
{ XTI_SNDLOWAT , " XTI_SNDLOWAT " } ,
{ 0 , NULL } ,
} ;
void
print_xti_optmgmt ( tcp , addr , len )
struct tcb * tcp ;
long addr ;
int len ;
{
int c = 0 ;
struct t_opthdr hdr ;
2002-05-24 14:19:44 +04:00
while ( len > = ( int ) sizeof hdr ) {
2002-05-23 15:48:58 +04:00
if ( umove ( tcp , addr , & hdr ) < 0 ) break ;
if ( c + + ) {
tprintf ( " , " ) ;
}
else if ( len > hdr . len + sizeof hdr ) {
tprintf ( " [ " ) ;
}
tprintf ( " {level= " ) ;
printxval ( xti_level , hdr . level , " ??? " ) ;
tprintf ( " , name= " ) ;
switch ( hdr . level ) {
case XTI_GENERIC :
printxval ( xti_generic , hdr . name , " XTI_??? " ) ;
break ;
default :
tprintf ( " %ld " , hdr . name ) ;
break ;
}
tprintf ( " , status= " ) ;
printxval ( transport_user_flags , hdr . status , " T_??? " ) ;
addr + = sizeof hdr ;
len - = sizeof hdr ;
if ( ( hdr . len - = sizeof hdr ) > 0 ) {
if ( hdr . len > len ) break ;
tprintf ( " , val= " ) ;
if ( len = = sizeof ( int ) )
printnum ( tcp , addr , " %d " ) ;
else
printstr ( tcp , addr , hdr . len ) ;
addr + = hdr . len ;
len - = hdr . len ;
}
tprintf ( " } " ) ;
}
if ( len > 0 ) {
if ( c + + ) tprintf ( " , " ) ;
printstr ( tcp , addr , len ) ;
}
if ( c > 1 ) tprintf ( " ] " ) ;
}
# endif
static void
print_optmgmt ( tcp , addr , len )
struct tcb * tcp ;
long addr ;
int len ;
{
/* We don't know how to tell if TLI (socket) or XTI
optmgmt is being used yet , assume TLI . */
# if defined (HAVE_OPTHDR)
print_sock_optmgmt ( tcp , addr , len ) ;
# elif defined (HAVE_T_OPTHDR)
print_xti_optmgmt ( tcp , addr , len ) ;
# else
printstr ( tcp , addr , len ) ;
# endif
}
static struct xlat service_type [ ] = {
{ T_COTS , " T_COTS " } ,
{ T_COTS_ORD , " T_COTS_ORD " } ,
{ T_CLTS , " T_CLTS " } ,
{ 0 , NULL } ,
} ;
static struct xlat ts_state [ ] = {
{ TS_UNBND , " TS_UNBND " } ,
{ TS_WACK_BREQ , " TS_WACK_BREQ " } ,
{ TS_WACK_UREQ , " TS_WACK_UREQ " } ,
{ TS_IDLE , " TS_IDLE " } ,
{ TS_WACK_OPTREQ , " TS_WACK_OPTREQ " } ,
{ TS_WACK_CREQ , " TS_WACK_CREQ " } ,
{ TS_WCON_CREQ , " TS_WCON_CREQ " } ,
{ TS_WRES_CIND , " TS_WRES_CIND " } ,
{ TS_WACK_CRES , " TS_WACK_CRES " } ,
{ TS_DATA_XFER , " TS_DATA_XFER " } ,
{ TS_WIND_ORDREL , " TS_WIND_ORDREL " } ,
{ TS_WREQ_ORDREL , " TS_WREQ_ORDREL " } ,
{ TS_WACK_DREQ6 , " TS_WACK_DREQ6 " } ,
{ TS_WACK_DREQ7 , " TS_WACK_DREQ7 " } ,
{ TS_WACK_DREQ9 , " TS_WACK_DREQ9 " } ,
{ TS_WACK_DREQ10 , " TS_WACK_DREQ10 " } ,
{ TS_WACK_DREQ11 , " TS_WACK_DREQ11 " } ,
{ 0 , NULL } ,
} ;
static struct xlat provider_flags [ ] = {
{ 0 , " 0 " } ,
{ SENDZERO , " SENDZERO " } ,
{ EXPINLINE , " EXPINLINE " } ,
{ XPG4_1 , " XPG4_1 " } ,
{ 0 , NULL } ,
} ;
static struct xlat tli_errors [ ] = {
{ TBADADDR , " TBADADDR " } ,
{ TBADOPT , " TBADOPT " } ,
{ TACCES , " TACCES " } ,
{ TBADF , " TBADF " } ,
{ TNOADDR , " TNOADDR " } ,
{ TOUTSTATE , " TOUTSTATE " } ,
{ TBADSEQ , " TBADSEQ " } ,
{ TSYSERR , " TSYSERR " } ,
{ TLOOK , " TLOOK " } ,
{ TBADDATA , " TBADDATA " } ,
{ TBUFOVFLW , " TBUFOVFLW " } ,
{ TFLOW , " TFLOW " } ,
{ TNODATA , " TNODATA " } ,
{ TNODIS , " TNODIS " } ,
{ TNOUDERR , " TNOUDERR " } ,
{ TBADFLAG , " TBADFLAG " } ,
{ TNOREL , " TNOREL " } ,
{ TNOTSUPPORT , " TNOTSUPPORT " } ,
{ TSTATECHNG , " TSTATECHNG " } ,
{ TNOSTRUCTYPE , " TNOSTRUCTYPE " } ,
{ TBADNAME , " TBADNAME " } ,
{ TBADQLEN , " TBADQLEN " } ,
{ TADDRBUSY , " TADDRBUSY " } ,
{ TINDOUT , " TINDOUT " } ,
{ TPROVMISMATCH , " TPROVMISMATCH " } ,
{ TRESQLEN , " TRESQLEN " } ,
{ TRESADDR , " TRESADDR " } ,
{ TQFULL , " TQFULL " } ,
{ TPROTO , " TPROTO " } ,
{ 0 , NULL } ,
} ;
static int
print_transport_message ( tcp , expect , addr , len )
struct tcb * tcp ;
int expect ;
long addr ;
int len ;
{
union T_primitives m ;
int c = 0 ;
if ( len < sizeof m . type ) goto dump ;
if ( umove ( tcp , addr , & m . type ) < 0 ) goto dump ;
# define GET(type, struct) \
do { \
if ( len < sizeof m . struct ) goto dump ; \
if ( umove ( tcp , addr , & m . struct ) < 0 ) goto dump ; \
tprintf ( " { " ) ; \
if ( expect ! = type ) { \
+ + c ; \
tprintf ( # type ) ; \
} \
} \
while ( 0 )
# define COMMA() \
do { if ( c + + ) tprintf ( " , " ) ; } while ( 0 )
# define STRUCT(struct, elem, print) \
do { \
COMMA ( ) ; \
if ( m . struct . elem # # _length < 0 | | \
m . struct . elem # # _offset < sizeof m . struct | | \
m . struct . elem # # _offset + m . struct . elem # # _length > len ) \
{ \
tprintf ( # elem " _length=%ld, " # elem " _offset=%ld " , \
m . struct . elem # # _length , \
m . struct . elem # # _offset ) ; \
} \
else { \
tprintf ( # elem " = " ) ; \
print ( tcp , \
addr + m . struct . elem # # _offset , \
m . struct . elem # # _length ) ; \
} \
} \
while ( 0 )
# define ADDR(struct, elem) STRUCT (struct, elem, printstr)
switch ( m . type ) {
# ifdef T_CONN_REQ
case T_CONN_REQ : /* connect request */
GET ( T_CONN_REQ , conn_req ) ;
ADDR ( conn_req , DEST ) ;
ADDR ( conn_req , OPT ) ;
break ;
# endif
# ifdef T_CONN_RES
case T_CONN_RES : /* connect response */
GET ( T_CONN_RES , conn_res ) ;
COMMA ( ) ;
tprintf ( " QUEUE=%p " , m . conn_res . QUEUE_ptr ) ;
ADDR ( conn_res , OPT ) ;
COMMA ( ) ;
tprintf ( " SEQ=%ld " , m . conn_res . SEQ_number ) ;
break ;
# endif
# ifdef T_DISCON_REQ
case T_DISCON_REQ : /* disconnect request */
GET ( T_DISCON_REQ , discon_req ) ;
COMMA ( ) ;
tprintf ( " SEQ=%ld " , m . discon_req . SEQ_number ) ;
break ;
# endif
# ifdef T_DATA_REQ
case T_DATA_REQ : /* data request */
GET ( T_DATA_REQ , data_req ) ;
COMMA ( ) ;
tprintf ( " MORE=%ld " , m . data_req . MORE_flag ) ;
break ;
# endif
# ifdef T_EXDATA_REQ
case T_EXDATA_REQ : /* expedited data req */
GET ( T_EXDATA_REQ , exdata_req ) ;
COMMA ( ) ;
tprintf ( " MORE=%ld " , m . exdata_req . MORE_flag ) ;
break ;
# endif
# ifdef T_INFO_REQ
case T_INFO_REQ : /* information req */
GET ( T_INFO_REQ , info_req ) ;
break ;
# endif
# ifdef T_BIND_REQ
case T_BIND_REQ : /* bind request */
# ifdef O_T_BIND_REQ
case O_T_BIND_REQ : /* Ugly xti/tli hack */
# endif
GET ( T_BIND_REQ , bind_req ) ;
ADDR ( bind_req , ADDR ) ;
COMMA ( ) ;
tprintf ( " CONIND=%ld " , m . bind_req . CONIND_number ) ;
break ;
# endif
# ifdef T_UNBIND_REQ
case T_UNBIND_REQ : /* unbind request */
GET ( T_UNBIND_REQ , unbind_req ) ;
break ;
# endif
# ifdef T_UNITDATA_REQ
case T_UNITDATA_REQ : /* unitdata requset */
GET ( T_UNITDATA_REQ , unitdata_req ) ;
ADDR ( unitdata_req , DEST ) ;
ADDR ( unitdata_req , OPT ) ;
break ;
# endif
# ifdef T_OPTMGMT_REQ
case T_OPTMGMT_REQ : /* manage opt req */
GET ( T_OPTMGMT_REQ , optmgmt_req ) ;
COMMA ( ) ;
tprintf ( " MGMT= " ) ;
printflags ( transport_user_flags , m . optmgmt_req . MGMT_flags ) ;
STRUCT ( optmgmt_req , OPT , print_optmgmt ) ;
break ;
# endif
# ifdef T_ORDREL_REQ
case T_ORDREL_REQ : /* orderly rel req */
GET ( T_ORDREL_REQ , ordrel_req ) ;
break ;
# endif
# ifdef T_CONN_IND
case T_CONN_IND : /* connect indication */
GET ( T_CONN_IND , conn_ind ) ;
ADDR ( conn_ind , SRC ) ;
ADDR ( conn_ind , OPT ) ;
tprintf ( " , SEQ=%ld " , m . conn_ind . SEQ_number ) ;
break ;
# endif
# ifdef T_CONN_CON
case T_CONN_CON : /* connect corfirm */
GET ( T_CONN_CON , conn_con ) ;
ADDR ( conn_con , RES ) ;
ADDR ( conn_con , OPT ) ;
break ;
# endif
# ifdef T_DISCON_IND
case T_DISCON_IND : /* discon indication */
GET ( T_DISCON_IND , discon_ind ) ;
COMMA ( ) ;
tprintf ( " DISCON=%ld, SEQ=%ld " ,
m . discon_ind . DISCON_reason , m . discon_ind . SEQ_number ) ;
break ;
# endif
# ifdef T_DATA_IND
case T_DATA_IND : /* data indication */
GET ( T_DATA_IND , data_ind ) ;
COMMA ( ) ;
tprintf ( " MORE=%ld " , m . data_ind . MORE_flag ) ;
break ;
# endif
# ifdef T_EXDATA_IND
case T_EXDATA_IND : /* expedited data ind */
GET ( T_EXDATA_IND , exdata_ind ) ;
COMMA ( ) ;
tprintf ( " MORE=%ld " , m . exdata_ind . MORE_flag ) ;
break ;
# endif
# ifdef T_INFO_ACK
case T_INFO_ACK : /* info ack */
GET ( T_INFO_ACK , info_ack ) ;
COMMA ( ) ;
tprintf ( " TSDU=%ld, ETSDU=%ld, CDATA=%ld, DDATA=%ld, "
" ADDR=%ld, OPT=%ld, TIDU=%ld, SERV= " ,
m . info_ack . TSDU_size , m . info_ack . ETSDU_size ,
m . info_ack . CDATA_size , m . info_ack . DDATA_size ,
m . info_ack . ADDR_size , m . info_ack . OPT_size ,
m . info_ack . TIDU_size ) ;
printxval ( service_type , m . info_ack . SERV_type , " T_??? " ) ;
tprintf ( " , CURRENT= " ) ;
printxval ( ts_state , m . info_ack . CURRENT_state , " TS_??? " ) ;
tprintf ( " , PROVIDER= " ) ;
printflags ( provider_flags , m . info_ack . PROVIDER_flag ) ;
break ;
# endif
# ifdef T_BIND_ACK
case T_BIND_ACK : /* bind ack */
GET ( T_BIND_ACK , bind_ack ) ;
ADDR ( bind_ack , ADDR ) ;
tprintf ( " , CONIND=%ld " , m . bind_ack . CONIND_number ) ;
break ;
# endif
# ifdef T_ERROR_ACK
case T_ERROR_ACK : /* error ack */
GET ( T_ERROR_ACK , error_ack ) ;
COMMA ( ) ;
tprintf ( " ERROR= " ) ;
printxval ( transport_user_options ,
m . error_ack . ERROR_prim , " TI_??? " ) ;
tprintf ( " , TLI= " ) ;
printxval ( tli_errors , m . error_ack . TLI_error , " T??? " ) ;
tprintf ( " UNIX=%s " , strerror ( m . error_ack . UNIX_error ) ) ;
break ;
# endif
# ifdef T_OK_ACK
case T_OK_ACK : /* ok ack */
GET ( T_OK_ACK , ok_ack ) ;
COMMA ( ) ;
tprintf ( " CORRECT= " ) ;
printxval ( transport_user_options ,
m . ok_ack . CORRECT_prim , " TI_??? " ) ;
break ;
# endif
# ifdef T_UNITDATA_IND
case T_UNITDATA_IND : /* unitdata ind */
GET ( T_UNITDATA_IND , unitdata_ind ) ;
ADDR ( unitdata_ind , SRC ) ;
ADDR ( unitdata_ind , OPT ) ;
break ;
# endif
# ifdef T_UDERROR_IND
case T_UDERROR_IND : /* unitdata error ind */
GET ( T_UDERROR_IND , uderror_ind ) ;
ADDR ( uderror_ind , DEST ) ;
ADDR ( uderror_ind , OPT ) ;
tprintf ( " , ERROR=%ld " , m . uderror_ind . ERROR_type ) ;
break ;
# endif
# ifdef T_OPTMGMT_ACK
case T_OPTMGMT_ACK : /* manage opt ack */
GET ( T_OPTMGMT_ACK , optmgmt_ack ) ;
COMMA ( ) ;
tprintf ( " MGMT= " ) ;
printflags ( transport_user_flags , m . optmgmt_ack . MGMT_flags ) ;
STRUCT ( optmgmt_ack , OPT , print_optmgmt ) ;
break ;
# endif
# ifdef T_ORDREL_IND
case T_ORDREL_IND : /* orderly rel ind */
GET ( T_ORDREL_IND , ordrel_ind ) ;
break ;
# endif
# ifdef T_ADDR_REQ
case T_ADDR_REQ : /* address req */
GET ( T_ADDR_REQ , addr_req ) ;
break ;
# endif
# ifdef T_ADDR_ACK
case T_ADDR_ACK : /* address response */
GET ( T_ADDR_ACK , addr_ack ) ;
ADDR ( addr_ack , LOCADDR ) ;
ADDR ( addr_ack , REMADDR ) ;
break ;
# endif
default :
dump :
c = - 1 ;
printstr ( tcp , addr , len ) ;
break ;
}
if ( c > = 0 ) tprintf ( " } " ) ;
# undef ADDR
# undef COMMA
# undef STRUCT
return 0 ;
}
1999-02-19 03:21:36 +03:00
# endif /* TI_BIND */
2002-05-23 15:48:58 +04:00
1999-02-19 03:21:36 +03:00
static int
internal_stream_ioctl ( tcp , arg )
struct tcb * tcp ;
int arg ;
{
struct strioctl si ;
char * name ;
int in_and_out ;
2002-05-23 15:48:58 +04:00
int timod = 0 ;
1999-02-19 03:21:36 +03:00
# ifdef SI_GETUDATA
struct si_udata udata ;
# endif /* SI_GETUDATA */
if ( ! arg )
return 0 ;
if ( umove ( tcp , arg , & si ) < 0 ) {
if ( entering ( tcp ) )
tprintf ( " , {...} " ) ;
return 1 ;
}
if ( entering ( tcp ) ) {
name = ioctl_lookup ( si . ic_cmd ) ;
if ( name )
tprintf ( " , {ic_cmd=%s " , name ) ;
else
tprintf ( " , {ic_cmd=%#x " , si . ic_cmd ) ;
if ( si . ic_timout = = INFTIM )
tprintf ( " , ic_timout=INFTIM, " ) ;
else
tprintf ( " ic_timout=%d, " , si . ic_timout ) ;
}
in_and_out = 1 ;
switch ( si . ic_cmd ) {
# ifdef SI_GETUDATA
case SI_GETUDATA :
in_and_out = 0 ;
break ;
# endif /* SI_GETUDATA */
}
if ( in_and_out ) {
if ( entering ( tcp ) )
tprintf ( " /* in */ " ) ;
else
tprintf ( " , /* out */ " ) ;
}
if ( in_and_out | | entering ( tcp ) )
tprintf ( " ic_len=%d, ic_dp= " , si . ic_len ) ;
switch ( si . ic_cmd ) {
# ifdef TI_BIND
case TI_BIND :
/* in T_BIND_REQ, out T_BIND_ACK */
2002-05-23 15:48:58 +04:00
+ + timod ;
1999-02-19 03:21:36 +03:00
if ( entering ( tcp ) ) {
2002-05-23 15:48:58 +04:00
print_transport_message ( tcp ,
T_BIND_REQ ,
si . ic_dp , si . ic_len ) ;
1999-02-19 03:21:36 +03:00
}
else {
2002-05-23 15:48:58 +04:00
print_transport_message ( tcp ,
T_BIND_ACK ,
si . ic_dp , si . ic_len ) ;
1999-02-19 03:21:36 +03:00
}
break ;
# endif /* TI_BIND */
# ifdef TI_UNBIND
case TI_UNBIND :
/* in T_UNBIND_REQ, out T_OK_ACK */
2002-05-23 15:48:58 +04:00
+ + timod ;
if ( entering ( tcp ) ) {
print_transport_message ( tcp ,
T_UNBIND_REQ ,
si . ic_dp , si . ic_len ) ;
}
else {
print_transport_message ( tcp ,
T_OK_ACK ,
si . ic_dp , si . ic_len ) ;
}
1999-02-19 03:21:36 +03:00
break ;
# endif /* TI_UNBIND */
# ifdef TI_GETINFO
case TI_GETINFO :
/* in T_INFO_REQ, out T_INFO_ACK */
2002-05-23 15:48:58 +04:00
+ + timod ;
if ( entering ( tcp ) ) {
print_transport_message ( tcp ,
T_INFO_REQ ,
si . ic_dp , si . ic_len ) ;
}
else {
print_transport_message ( tcp ,
T_INFO_ACK ,
si . ic_dp , si . ic_len ) ;
}
1999-02-19 03:21:36 +03:00
break ;
# endif /* TI_GETINFO */
# ifdef TI_OPTMGMT
case TI_OPTMGMT :
/* in T_OPTMGMT_REQ, out T_OPTMGMT_ACK */
2002-05-23 15:48:58 +04:00
+ + timod ;
if ( entering ( tcp ) ) {
print_transport_message ( tcp ,
T_OPTMGMT_REQ ,
si . ic_dp , si . ic_len ) ;
}
else {
print_transport_message ( tcp ,
T_OPTMGMT_ACK ,
si . ic_dp , si . ic_len ) ;
}
1999-02-19 03:21:36 +03:00
break ;
# endif /* TI_OPTMGMT */
# ifdef SI_GETUDATA
case SI_GETUDATA :
if ( entering ( tcp ) )
break ;
#if 0
tprintf ( " struct si_udata " ) ;
# endif
if ( umove ( tcp , ( int ) si . ic_dp , & udata ) < 0 )
tprintf ( " {...} " ) ;
else {
tprintf ( " {tidusize=%d, addrsize=%d, " ,
udata . tidusize , udata . addrsize ) ;
tprintf ( " optsize=%d, etsdusize=%d, " ,
udata . optsize , udata . etsdusize ) ;
tprintf ( " servtype=%d, so_state=%d, " ,
udata . servtype , udata . so_state ) ;
tprintf ( " so_options=%d " , udata . so_options ) ;
#if 0
tprintf ( " , tsdusize=%d " , udata . tsdusize ) ;
# endif
tprintf ( " } " ) ;
}
break ;
# endif /* SI_GETUDATA */
default :
2002-10-07 18:31:00 +04:00
printstr ( tcp , ( long ) si . ic_dp , si . ic_len ) ;
1999-02-19 03:21:36 +03:00
break ;
}
2002-05-23 15:48:58 +04:00
if ( exiting ( tcp ) ) {
1999-02-19 03:21:36 +03:00
tprintf ( " } " ) ;
2002-05-23 15:48:58 +04:00
if ( timod & & tcp - > u_rval ) {
tcp - > auxstr = xlookup ( tli_errors , tcp - > u_rval ) ;
return RVAL_STR + 1 ;
}
}
1999-02-19 03:21:36 +03:00
return 1 ;
}
int
stream_ioctl ( tcp , code , arg )
struct tcb * tcp ;
int code , arg ;
{
# ifdef I_LIST
int i ;
# endif
int val ;
# ifdef I_FLUSHBAND
struct bandinfo bi ;
# endif
struct strpeek sp ;
struct strfdinsert sfi ;
struct strrecvfd srf ;
# ifdef I_LIST
struct str_list sl ;
# endif
/* I_STR is a special case because the data is read & written. */
if ( code = = I_STR )
return internal_stream_ioctl ( tcp , arg ) ;
if ( entering ( tcp ) )
return 0 ;
switch ( code ) {
case I_PUSH :
case I_LOOK :
case I_FIND :
/* arg is a string */
tprintf ( " , " ) ;
printpath ( tcp , arg ) ;
return 1 ;
case I_POP :
/* doesn't take an argument */
return 1 ;
case I_FLUSH :
/* argument is an option */
tprintf ( " , " ) ;
printxval ( stream_flush_options , arg , " FLUSH??? " ) ;
return 1 ;
# ifdef I_FLUSHBAND
case I_FLUSHBAND :
/* argument is a pointer to a bandinfo struct */
if ( umove ( tcp , arg , & bi ) < 0 )
tprintf ( " , {...} " ) ;
else {
tprintf ( " , {bi_pri=%d, bi_flag= " , bi . bi_pri ) ;
if ( ! printflags ( stream_flush_options , bi . bi_flag ) )
tprintf ( " 0 " ) ;
tprintf ( " } " ) ;
}
return 1 ;
# endif /* I_FLUSHBAND */
case I_SETSIG :
/* argument is a set of flags */
tprintf ( " , " ) ;
if ( ! printflags ( stream_setsig_flags , arg ) )
tprintf ( " 0 " ) ;
return 1 ;
case I_GETSIG :
/* argument is a pointer to a set of flags */
if ( syserror ( tcp ) )
return 0 ;
tprintf ( " , [ " ) ;
if ( umove ( tcp , arg , & val ) < 0 )
tprintf ( " ? " ) ;
else if ( ! printflags ( stream_setsig_flags , val ) )
tprintf ( " 0 " ) ;
tprintf ( " ] " ) ;
return 1 ;
case I_PEEK :
/* argument is a pointer to a strpeek structure */
if ( syserror ( tcp ) | | ! arg )
return 0 ;
if ( umove ( tcp , arg , & sp ) < 0 ) {
tprintf ( " , {...} " ) ;
return 1 ;
}
tprintf ( " , {ctlbuf= " ) ;
printstrbuf ( tcp , & sp . ctlbuf , 1 ) ;
tprintf ( " , databuf= " ) ;
printstrbuf ( tcp , & sp . databuf , 1 ) ;
2002-05-17 15:41:35 +04:00
tprintf ( " , flags= " ) ;
1999-02-19 03:21:36 +03:00
if ( ! printflags ( msgflags , sp . flags ) )
tprintf ( " 0 " ) ;
2002-05-17 15:41:35 +04:00
tprintf ( " } " ) ;
1999-02-19 03:21:36 +03:00
return 1 ;
case I_SRDOPT :
/* argument is an option with flags */
tprintf ( " , " ) ;
printxval ( stream_read_options , arg & RMODEMASK , " R??? " ) ;
addflags ( stream_read_flags , arg & ~ RMODEMASK ) ;
return 1 ;
case I_GRDOPT :
/* argument is an pointer to an option with flags */
if ( syserror ( tcp ) )
return 0 ;
tprintf ( " , [ " ) ;
if ( umove ( tcp , arg , & val ) < 0 )
tprintf ( " ? " ) ;
else {
printxval ( stream_read_options ,
arg & RMODEMASK , " R??? " ) ;
addflags ( stream_read_flags , arg & ~ RMODEMASK ) ;
}
tprintf ( " ] " ) ;
return 1 ;
case I_NREAD :
# ifdef I_GETBAND
case I_GETBAND :
# endif
# ifdef I_SETCLTIME
case I_SETCLTIME :
# endif
# ifdef I_GETCLTIME
case I_GETCLTIME :
# endif
/* argument is a pointer to a decimal integer */
if ( syserror ( tcp ) )
return 0 ;
tprintf ( " , " ) ;
printnum ( tcp , arg , " %d " ) ;
return 1 ;
case I_FDINSERT :
/* argument is a pointer to a strfdinsert structure */
if ( syserror ( tcp ) | | ! arg )
return 0 ;
if ( umove ( tcp , arg , & sfi ) < 0 ) {
tprintf ( " , {...} " ) ;
return 1 ;
}
tprintf ( " , {ctlbuf= " ) ;
printstrbuf ( tcp , & sfi . ctlbuf , 1 ) ;
tprintf ( " , databuf= " ) ;
printstrbuf ( tcp , & sfi . databuf , 1 ) ;
2002-05-17 15:41:35 +04:00
tprintf ( " , flags= " ) ;
1999-02-19 03:21:36 +03:00
if ( ! printflags ( msgflags , sfi . flags ) )
tprintf ( " 0 " ) ;
tprintf ( " , filedes=%d, offset=%d} " , sfi . fildes , sfi . offset ) ;
return 1 ;
# ifdef I_SWROPT
case I_SWROPT :
/* argument is a set of flags */
tprintf ( " , " ) ;
if ( ! printflags ( stream_write_flags , arg ) )
tprintf ( " 0 " ) ;
return 1 ;
# endif /* I_SWROPT */
# ifdef I_GWROPT
case I_GWROPT :
/* argument is an pointer to an option with flags */
if ( syserror ( tcp ) )
return 0 ;
tprintf ( " , [ " ) ;
if ( umove ( tcp , arg , & val ) < 0 )
tprintf ( " ? " ) ;
else if ( ! printflags ( stream_write_flags , arg ) )
tprintf ( " 0 " ) ;
tprintf ( " ] " ) ;
return 1 ;
# endif /* I_GWROPT */
case I_SENDFD :
# ifdef I_CKBAND
case I_CKBAND :
# endif
# ifdef I_CANPUT
case I_CANPUT :
# endif
case I_LINK :
case I_UNLINK :
case I_PLINK :
case I_PUNLINK :
/* argument is a decimal integer */
tprintf ( " , %d " , arg ) ;
return 1 ;
case I_RECVFD :
/* argument is a pointer to a strrecvfd structure */
if ( syserror ( tcp ) | | ! arg )
return 0 ;
if ( umove ( tcp , arg , & srf ) < 0 ) {
tprintf ( " , {...} " ) ;
return 1 ;
}
tprintf ( " , {fd=%d, uid=%lu, gid=%lu} " , srf . fd ,
( unsigned long ) srf . uid , ( unsigned long ) srf . gid ) ;
return 1 ;
# ifdef I_LIST
case I_LIST :
if ( syserror ( tcp ) )
return 0 ;
if ( arg = = 0 ) {
tprintf ( " , NULL " ) ;
return 1 ;
}
if ( umove ( tcp , arg , & sl ) < 0 ) {
tprintf ( " , {...} " ) ;
return 1 ;
}
tprintf ( " , {sl_nmods=%d, sl_modlist=[ " , sl . sl_nmods ) ;
for ( i = 0 ; i < tcp - > u_rval ; i + + ) {
if ( i )
tprintf ( " , " ) ;
printpath ( tcp , ( int ) sl . sl_modlist [ i ] . l_name ) ;
}
tprintf ( " ]} " ) ;
return 1 ;
# endif /* I_LIST */
# ifdef I_ATMARK
case I_ATMARK :
tprintf ( " , " ) ;
printxval ( stream_atmark_options , arg , " ???MARK " ) ;
return 1 ;
# endif /* I_ATMARK */
default :
return 0 ;
}
}
2000-09-02 01:03:06 +04:00
# endif /* !linux && !FREEBSD */
1999-02-19 03:21:36 +03:00
2000-09-02 01:03:06 +04:00
# endif /* HAVE_SYS_STREAM_H || linux || FREEBSD */
1999-02-19 03:21:36 +03:00