NT4/private/ntos/tdi/isn/ipxroute/ipxroute.c

1823 lines
48 KiB
C
Raw Normal View History

2001-01-01 00:00:00 +01:00
/****************************************************************************
* (c) Copyright 1990, 1993 Micro Computer Systems, Inc. All rights reserved.
*****************************************************************************
*
* Title: IPX/SPX Compatible Source Routing Daemon for Windows NT
*
* Module: ipx/route/ipxroute.c
*
* Version: 1.00.00
*
* Date: 04-08-93
*
* Author: Brian Walker
*
*****************************************************************************
*
* Change Log:
*
* Date DevSFC Comment
* -------- ------ -------------------------------------------------------
* 02-14-95 RamC Added command line options to support displaying
* Router Table, Router Statistics and SAP information.
* Basically a merge of ipxroute and Stefan's rttest.
*****************************************************************************
*
* Functional Description:
*
*
****************************************************************************/
#include <nt.h>
#include <ntrtl.h>
#include <nturtl.h>
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <process.h>
#include <ntstapi.h>
#include <sys/stropts.h>
#include <windows.h>
#include "errno.h"
#include "tdi.h"
#include "isnkrnl.h"
#include "ipxrtmsg.h"
#include "..\rip\driver.h"
#include "utils.h"
#include "nwsap.h"
typedef struct _IPX_ROUTE_ENTRY {
UCHAR Network[4];
USHORT NicId;
UCHAR NextRouter[6];
PVOID NdisBindingContext;
USHORT Flags;
USHORT Timer;
UINT Segment;
USHORT TickCount;
USHORT HopCount;
PVOID AlternateRoute[2];
PVOID NicLinkage[2];
struct {
PVOID Linkage[2];
ULONG Reserved[1];
} PRIVATE;
} IPX_ROUTE_ENTRY, * PIPX_ROUTE_ENTRY;
IPX_ROUTE_ENTRY rte;
/** Global Variables **/
int sr_def = 0;
int sr_bcast = 0;
int sr_multi = 0;
int boardnum = 0;
int clear = 0;
int config = 0;
int showtable = 0;
int showservers = 0;
int showstats = 0;
int clearstats = 0;
int servertype;
char nodeaddr[6]; /* Node address to remove */
HANDLE nwlinkfd;
HANDLE isnipxfd;
HANDLE isnripfd;
char ebuffer[128];
char nwlinkname[] = "\\Device\\Streams\\NWLinkIpx";
wchar_t isnipxname[] = L"\\Device\\NwlnkIpx";
wchar_t isnripname[] = L"\\Device\\Ipxroute";
char pgmname[] = "IPXROUTE";
#define SHOW_ALL_SERVERS 0XFFFF
/** **/
#define INVALID_HANDLE (HANDLE)(-1)
/** Structure to send REMOVE with **/
typedef struct rterem {
int rterem_bnum; /* Board number */
char rterem_node[6]; /* Node to remove */
} rterem;
typedef int (_CRTAPI1 * PQSORT_COMPARE)(const void * p0, const void * p1);
/** Internal Function Prototypes **/
extern void print_table(int);
extern void usage(void);
extern void print_version(void);
extern char *print_type(int);
extern int my_strncmp(char *, char *, int);
extern int get_board_num(char *, int *);
extern int get_node_num(char *, char *);
extern int get_server_type(char *, int *);
extern unsigned char get_hex_byte(char *);
extern int get_driver_parms(void);
extern int set_driver_parms(void);
extern int do_strioctl(HANDLE, int, char *, int, int);
extern void remove_address(char *);
extern void clear_table(void);
extern void print_config(void);
extern unsigned long get_emsg(int);
int do_isnipxioctl(HANDLE fd, int cmd, char *datap, int dlen);
unsigned long put_msg(BOOLEAN error, unsigned long MsgNum, ... );
char *load_msg( unsigned long MsgNum, ... );
extern void show_router_table(PHANDLE, PIO_STATUS_BLOCK);
extern void show_stats(HANDLE, PIO_STATUS_BLOCK);
extern void clear_stats(HANDLE, PIO_STATUS_BLOCK);
extern void show_servers(int);
extern int _CRTAPI1 CompareServerNames( void * p0, void * p1);
extern int _CRTAPI1 CompareNetNumber( void * p0, void * p1);
/*page*************************************************************
m a i n
This is the main routine that gets executed when a NET START
happens.
Arguments - None
Returns - Nothing
********************************************************************/
void _CRTAPI1 main(int argc, char **argv)
{
char *p;
int todo;
int remove_flag;
UNICODE_STRING FileString;
OBJECT_ATTRIBUTES ObjectAttributes, RouterObjectAttributes;
IO_STATUS_BLOCK IoStatusBlock, RouterIoStatusBlock;
NTSTATUS Status;
/** **/
print_version();
/** Open the nwlink driver **/
nwlinkfd = s_open(nwlinkname, 0, 0);
/** Open the isnipx driver **/
RtlInitUnicodeString (&FileString, isnipxname);
InitializeObjectAttributes(
&ObjectAttributes,
&FileString,
OBJ_CASE_INSENSITIVE,
NULL,
NULL);
Status = NtOpenFile(
&isnipxfd,
SYNCHRONIZE | FILE_READ_DATA | FILE_WRITE_DATA,
&ObjectAttributes,
&IoStatusBlock,
FILE_SHARE_READ | FILE_SHARE_WRITE,
FILE_SYNCHRONOUS_IO_ALERT);
if (!NT_SUCCESS(Status)) {
isnipxfd = INVALID_HANDLE;
put_msg (TRUE, MSG_OPEN_FAILED, "\\Device\\NwlnkIpx");
}
/** Open the isnrip driver **/
RtlInitUnicodeString (&FileString, isnripname);
InitializeObjectAttributes(
&RouterObjectAttributes,
&FileString,
OBJ_CASE_INSENSITIVE,
NULL,
NULL);
Status = NtOpenFile(
&isnripfd,
SYNCHRONIZE | GENERIC_READ,
&RouterObjectAttributes,
&RouterIoStatusBlock,
FILE_SHARE_READ | FILE_SHARE_WRITE,
FILE_SYNCHRONOUS_IO_NONALERT);
if (!NT_SUCCESS(Status)) {
isnripfd = INVALID_HANDLE;
// don't display any error message, but display the
// message that the IPX router is not started when
// the user actually tries to look at the router.
}
if ((nwlinkfd == INVALID_HANDLE) &&
(isnipxfd == INVALID_HANDLE) &&
(isnripfd == INVALID_HANDLE))
{
exit(1);
}
/** Go thru the command line and set it up **/
argc--;
argv++;
/** Parse the command line **/
todo = 0;
remove_flag = 0;
while (argc--) {
/** Uppercase the arg **/
p = *argv;
_strupr(p);
/** Parse the argument **/
if (!strcmp(p, "CLEAR")) {
todo = 1;
clear = 1;
}
else if (!strcmp(p, "DEF")) {
todo = 1;
sr_def = 1;
}
else if (!strcmp(p, "GBR")) {
todo = 1;
sr_bcast = 1;
}
else if (!strcmp(p, "MBR")) {
todo = 1;
sr_multi = 1;
}
else if (!strcmp(p, "CONFIG")) {
todo = 1;
config = 1;
}
else if (!my_strncmp(p, "BOARD=", 6))
get_board_num(p + 6, &boardnum);
else if (!my_strncmp(p, "REMOVE=", 7)) {
remove_flag = 1;
get_node_num(p + 7, nodeaddr);
}
else if (!strcmp(p, "TABLE")) {
todo = 1;
showtable = 1;
}
else if (!strcmp(p, "SERVERS")) {
todo = 1;
showservers = 1;
/** default is to show all server types **/
servertype = SHOW_ALL_SERVERS;
argv++;
if(argc--) {
p = *argv;
_strupr(p);
if (!my_strncmp(p, "/TYPE=", 6)) {
get_server_type(p + 6, &servertype);
}
else
usage();
}
/** no more arguments - break out of while lop **/
else
break;
}
else if (!strcmp(p, "STATS")) {
todo = 1;
/** default is to show the router statistics **/
showstats = 1;
argv++;
if(argc--) {
p = *argv;
_strupr(p);
if (!strcmp(p, "/CLEAR")) {
clearstats = 1;
showstats = 0;
}
else if (!strcmp(p, "/SHOW")) {
showstats = 1;
}
else
usage;
}
/** no more arguments - break out of while lop **/
else
break;
}
else
usage();
/** Goto the next entry **/
argv++;
}
/** Go update the driver **/
#if 0
printf("todo = %d\n", todo);
printf("remove_flag= %d\n", remove_flag);
printf("Clear flag = %d\n", clear);
printf("Config flag = %d\n", config);
printf("SR_DEF = %d\n", sr_def);
printf("SR_BCAST = %d\n", sr_bcast);
printf("SR_MULTI = %d\n", sr_multi);
printf("Board = %d\n", boardnum);
printf("Node = %02x:%02x:%02x:%02x:%02x:%02x\n",
(unsigned char)nodeaddr[0],
(unsigned char)nodeaddr[1],
(unsigned char)nodeaddr[2],
(unsigned char)nodeaddr[3],
(unsigned char)nodeaddr[4],
(unsigned char)nodeaddr[5]);
#endif
/** If we have a remove - go remove it and leave **/
if (remove_flag)
remove_address(nodeaddr); /* Does not return */
/** If clear - go clear the source routing table **/
if (clear)
clear_table(); /* Does not return */
/** If config - print out config **/
if (config)
print_config(); /* Does not return */
/** If showtable - print out routing table **/
if (showtable)
show_router_table(&isnripfd, &RouterIoStatusBlock); /* Does not return */
/** If showservers - print out selected servers list **/
if (showservers)
show_servers(servertype); /* Does not return */
/** If showstats - print out statistics **/
if (showstats)
show_stats(&isnripfd, &RouterIoStatusBlock); /* Does not return */
/** If clearstats - go clear statistics **/
if (clearstats)
clear_stats(&isnripfd, &RouterIoStatusBlock); /* Does not return */
/** If there is nothing to do - just print out everything **/
if (!todo) {
/** Get the driver parms **/
if (get_driver_parms()) {
if (isnipxfd != INVALID_HANDLE) NtClose(isnipxfd);
if (nwlinkfd != INVALID_HANDLE) NtClose(nwlinkfd);
if (isnripfd != INVALID_HANDLE) NtClose(isnripfd);
exit(1);
}
/** Print out the table (Never comes back) **/
print_table(1);
}
/** Go set the parameters **/
set_driver_parms();
/** Print the table out **/
print_table(0);
/** All Done **/
if (isnipxfd != INVALID_HANDLE) NtClose(isnipxfd);
if (nwlinkfd != INVALID_HANDLE) NtClose(nwlinkfd);
if (isnripfd != INVALID_HANDLE) NtClose(isnripfd);
exit(0);
}
/*page*************************************************************
p r i n t _ t a b l e
Print out the status of the source routing.
Arguments - flag = 0 - Do NOT print the table
1 - Do print the table (Never returns)
Returns - Nothing
********************************************************************/
void print_table(int flag)
{
/** Print the information **/
char * ptype;
printf("\n");
ptype = print_type(sr_def);
put_msg (FALSE, MSG_DEFAULT_NODE, ptype);
LocalFree (ptype);
printf("\n");
ptype = print_type(sr_bcast);
put_msg (FALSE, MSG_BROADCAST, ptype);
LocalFree (ptype);
printf("\n");
ptype = print_type(sr_multi);
put_msg (FALSE, MSG_MULTICAST, ptype);
LocalFree (ptype);
printf("\n");
if (!flag)
return;
#if 0
printf("\n");
printf(" Node Address Source Route\n");
printf("\n");
#endif
/** All Done **/
if (isnipxfd != INVALID_HANDLE) NtClose(isnipxfd);
if (nwlinkfd != INVALID_HANDLE) NtClose(nwlinkfd);
if (isnripfd != INVALID_HANDLE) NtClose(isnripfd);
exit(0);
}
/*page*************************************************************
u s a g e
Print the usage message.
Arguments - None
Returns - Nothing
********************************************************************/
void usage(void)
{
put_msg( FALSE, MSG_USAGE, pgmname );
/** All Done **/
if (isnipxfd != INVALID_HANDLE) NtClose(isnipxfd);
if (nwlinkfd != INVALID_HANDLE) NtClose(nwlinkfd);
if (isnripfd != INVALID_HANDLE) NtClose(isnripfd);
exit(0);
}
/*page*************************************************************
p r i n t _ v e r s i o n
Print the version number
Arguments - None
Returns - Nothing
********************************************************************/
void print_version(void)
{
printf("\n");
put_msg (FALSE, MSG_VERSION);
return;
}
/*page*************************************************************
p r i n t _ t y p e
Returns the broadcast type given in a string, the caller
must free the string.
Arguments - 0 = SINGLE ROUTE
Else = ALL ROUTES
Returns - Nothing
********************************************************************/
char *print_type(int flag)
{
if (flag)
return load_msg (MSG_ALL_ROUTE);
else
return load_msg (MSG_SINGLE_ROUTE);
}
/*page*************************************************************
m y _ s t r n c m p
Given a string (p), see if the first len chars are the same
as those of the 2nd string (s).
Arguments - p = Ptr to first string
s = Ptr to 2nd string
len = Length to check
Returns - 0 = Matched
Else = Not matched
********************************************************************/
int my_strncmp(char *p, char *s, int len)
{
/** **/
while (len--) {
if (*p++ != *s++)
return 1;
}
/** They matched **/
return 0;
}
/*page*************************************************************
g e t _ b o a r d _ n u m
Get the decimal number from the command line
Arguments - p = Ptr to the ASCII number
nump = Store the number here
Returns - 0 = Got the number OK
Else = Bad number
********************************************************************/
int get_board_num(char *p, int *nump)
{
*nump = atoi(p);
return 0;
}
/*page*************************************************************
g e t _ n o d e _ n u m
Get a node address from the command line
Arguments - p = Ptr to the ASCII number
nodep = Store the node number here
Returns - 0 = Got the number OK
Else = Bad number
********************************************************************/
int get_node_num(char *p, char *nodep)
{
int i;
unsigned char c1;
unsigned char c2;
/** **/
if (strlen(p) != 12) {
put_msg (TRUE, MSG_INVALID_REMOVE);
exit(1);
}
/** Get the number **/
for (i = 0 ; i < 6 ; i++) {
/** Get the next 2 digits **/
c1 = get_hex_byte(p++);
c2 = get_hex_byte(p++);
/** If we got a bad number - return error **/
if ((c1 == 0xFF) || (c2 == 0xFF)) {
put_msg (TRUE, MSG_INVALID_REMOVE);
exit(1);
}
/** Set the next byte **/
*nodep++ = (c1 << 4) + c2;
}
/** Return OK **/
return 0;
}
/*page*************************************************************
g e t _ s e r v e r _ t y p e
Get the decimal number from the command line
Arguments - p = Ptr to the ASCII number
nump = Store the number here
Returns - 0 = Got the number OK
Else = Bad number
********************************************************************/
int get_server_type(char *p, int *nump)
{
*nump = atoi(p);
return 0;
}
/*page*************************************************************
g e t _ h e x _ b y t e
Take 1 ascii hex chars and convert to a hex byte
Arguments - p = Ptr to the ASCII number
Returns - The number
(0xFF = Error)
********************************************************************/
unsigned char get_hex_byte(char *p)
{
unsigned char c;
/** Get the char **/
c = *(unsigned char *)p;
/** If 0-9 handle it **/
if ((c >= '0') && (c <= '9')) {
c -= '0';
return c;
}
/** If A-F handle it **/
if ((c >= 'A') && (c <= 'F')) {
c -= ('A' - 10);
return c;
}
/** This is a bad number **/
return 0xFF;
}
/*page***************************************************************
g e t _ d r i v e r _ p a r m s
Get the parameters from the driver
Arguments - None
Returns - 0 = OK
else = Error
********************************************************************/
int get_driver_parms()
{
int rc;
int buffer[4];
/** Set the board number **/
buffer[0] = boardnum;
/** Get the parms **/
if (nwlinkfd != INVALID_HANDLE) {
rc = do_strioctl(nwlinkfd, MIPX_SRGETPARMS, (char *)buffer, 4*sizeof(int), 0);
if (rc) {
/** Get the error code **/
rc = GetLastError();
put_msg (TRUE, MSG_BAD_PARAMETERS, "nwlink");
put_msg (TRUE, get_emsg(rc));
/** Return the error **/
return rc;
}
}
if (isnipxfd != INVALID_HANDLE) {
rc = do_isnipxioctl(isnipxfd, MIPX_SRGETPARMS, (char *)buffer, 4*sizeof(int));
if (rc) {
put_msg (TRUE, MSG_BAD_PARAMETERS, "nwlnkipx");
put_msg (TRUE, get_emsg(rc));
/** Return the error **/
return rc;
}
}
/** Get the variables **/
sr_def = buffer[1];
sr_bcast = buffer[2];
sr_multi = buffer[3];
/** Return OK **/
return 0;
}
/*page***************************************************************
s e t _ d r i v e r _ p a r m s
Set the parameters for the driver
Arguments - None
Returns - 0 = OK
else = Error
********************************************************************/
int set_driver_parms()
{
int rc;
int buffer[2];
/** Set the DEFAULT parm **/
buffer[0] = boardnum;
buffer[1] = sr_def;
if (nwlinkfd != INVALID_HANDLE) {
rc = do_strioctl(nwlinkfd, MIPX_SRDEF, (char *)buffer, 2 * sizeof(int), 0);
if (rc) {
rc = GetLastError();
put_msg (TRUE, MSG_SET_DEFAULT_ERROR, "nwlink");
put_msg (TRUE, get_emsg(rc));
return rc;
}
}
if (isnipxfd != INVALID_HANDLE) {
rc = do_isnipxioctl(isnipxfd, MIPX_SRDEF, (char *)buffer, 2 * sizeof(int));
if (rc) {
put_msg (TRUE, MSG_SET_DEFAULT_ERROR, "nwlnkipx");
put_msg (TRUE, get_emsg(rc));
return rc;
}
}
/** Set the BROADCAST parm **/
buffer[0] = boardnum;
buffer[1] = sr_bcast;
if (nwlinkfd != INVALID_HANDLE) {
rc = do_strioctl(nwlinkfd, MIPX_SRBCAST, (char *)buffer, 2 * sizeof(int), 0);
if (rc) {
rc = GetLastError();
put_msg (TRUE, MSG_SET_BROADCAST_ERROR, "nwlink");
put_msg (TRUE, get_emsg(rc));
return rc;
}
}
if (isnipxfd != INVALID_HANDLE) {
rc = do_isnipxioctl(isnipxfd, MIPX_SRBCAST, (char *)buffer, 2 * sizeof(int));
if (rc) {
put_msg (TRUE, MSG_SET_BROADCAST_ERROR, "nwlnkipx");
put_msg (TRUE, get_emsg(rc));
return rc;
}
}
/** Set the MULTICAST parm **/
buffer[0] = boardnum;
buffer[1] = sr_multi;
if (nwlinkfd != INVALID_HANDLE) {
rc = do_strioctl(nwlinkfd, MIPX_SRMULTI, (char *)buffer, 2 * sizeof(int), 0);
if (rc) {
rc = GetLastError();
put_msg (TRUE, MSG_SET_MULTICAST_ERROR, "nwlink");
put_msg (TRUE, get_emsg(rc));
return rc;
}
}
if (isnipxfd != INVALID_HANDLE) {
rc = do_isnipxioctl(isnipxfd, MIPX_SRMULTI, (char *)buffer, 2 * sizeof(int));
if (rc) {
put_msg (TRUE, MSG_SET_MULTICAST_ERROR, "nwlnkipx");
put_msg (TRUE, get_emsg(rc));
return rc;
}
}
/** Return OK **/
return 0;
}
/*page***************************************************************
d o _ s t r i o c t l
Do a stream ioctl
Arguments - fd = Handle to put on
cmd = Command to send
datap = Ptr to ctrl buffer
dlen = Ptr to len of data buffer
timout = Timeout value
Returns - 0 = OK
else = Error
********************************************************************/
int do_strioctl(HANDLE fd, int cmd, char *datap, int dlen, int timout)
{
int rc;
struct strioctl io;
/** Fill out the structure **/
io.ic_cmd = cmd;
io.ic_dp = datap;
io.ic_len = dlen;
io.ic_timout = timout;
/** Issue the ioctl **/
rc = s_ioctl(fd, I_STR, &io);
/** All Done **/
return rc;
}
/*page***************************************************************
r e m o v e _ a d d r e s s
Remove an address from the source routing table.
Arguments - nodep = Ptr to node address to remove
Returns - Does not return
********************************************************************/
void remove_address(char *nodep)
{
int rc;
int len;
rterem buf;
/** Build the area to send down to the driver **/
buf.rterem_bnum = boardnum;
memcpy(buf.rterem_node, nodep, 6);
len = sizeof(int) + 6;
/** Send the ioctl to remove the address **/
if (nwlinkfd != INVALID_HANDLE) {
rc = do_strioctl(nwlinkfd, MIPX_SRREMOVE, (char *)&buf, len, 0);
if (rc) {
rc = GetLastError();
put_msg (TRUE, MSG_REMOVE_ADDRESS_ERROR, "nwlink");
put_msg (TRUE, get_emsg(rc));
}
}
if (isnipxfd != INVALID_HANDLE) {
rc = do_isnipxioctl(isnipxfd, MIPX_SRREMOVE, (char *)&buf, len);
if (rc) {
put_msg (TRUE, MSG_REMOVE_ADDRESS_ERROR, "nwlnkipx");
if (rc == EINVAL) {
put_msg (TRUE, MSG_ADDRESS_NOT_FOUND);
} else {
put_msg (TRUE, get_emsg(rc));
}
}
}
/** Close up and exit **/
if (isnipxfd != INVALID_HANDLE) NtClose(isnipxfd);
if (nwlinkfd != INVALID_HANDLE) NtClose(nwlinkfd);
if (isnripfd != INVALID_HANDLE) NtClose(isnripfd);
exit(0);
}
/*page***************************************************************
c l e a r _ t a b l e
Clear out the routing table
Arguments - None
Returns - Does not return
********************************************************************/
void clear_table(void)
{
int rc;
/** Send the ioctl to clear the table **/
if (nwlinkfd != INVALID_HANDLE) {
rc = do_strioctl(nwlinkfd, MIPX_SRCLEAR, (char *)&boardnum, sizeof(int), 0);
if (rc) {
rc= GetLastError();
put_msg (TRUE, MSG_CLEAR_TABLE_ERROR, "nwlink");
put_msg (TRUE, get_emsg(rc));
}
}
if (isnipxfd != INVALID_HANDLE) {
rc = do_isnipxioctl(isnipxfd, MIPX_SRCLEAR, (char *)&boardnum, sizeof(int));
if (rc) {
put_msg (TRUE, MSG_CLEAR_TABLE_ERROR, "nwlnkipx");
put_msg (TRUE, get_emsg(rc));
}
}
/** Close up and exit **/
if (isnipxfd != INVALID_HANDLE) NtClose(isnipxfd);
if (nwlinkfd != INVALID_HANDLE) NtClose(nwlinkfd);
if (isnripfd != INVALID_HANDLE) NtClose(isnripfd);
exit(0);
}
typedef struct _ISN_ACTION_GET_DETAILS {
USHORT NicId; // passed by caller
BOOLEAN BindingSet; // returns TRUE if in set
UCHAR Type; // 1 = lan, 2 = up wan, 3 = down wan
ULONG FrameType; // returns 0 through 3
ULONG NetworkNumber; // returns virtual net if NicId is 0
UCHAR Node[6]; // adapter's MAC address.
WCHAR AdapterName[64]; // terminated with Unicode NULL
} ISN_ACTION_GET_DETAILS, *PISN_ACTION_GET_DETAILS;
#define REORDER_ULONG(_Ulong) \
((((_Ulong) & 0xff000000) >> 24) | \
(((_Ulong) & 0x00ff0000) >> 8) | \
(((_Ulong) & 0x0000ff00) << 8) | \
(((_Ulong) & 0x000000ff) << 24))
/*page***************************************************************
p r i n t _ c o n f i g
Prints out the current config
Arguments - None
Returns - Does not return
********************************************************************/
void print_config(void)
{
int rc;
int nicid, nicidcount;
int showlegend = 0;
char nicidbuf[6];
char network[9];
char * frametype;
char node[13];
char special[2];
char adaptername[64];
ISN_ACTION_GET_DETAILS getdetails;
if (isnipxfd != INVALID_HANDLE) {
/** First query nicid 0 **/
getdetails.NicId = 0;
rc = do_isnipxioctl(isnipxfd, MIPX_CONFIG, (char *)&getdetails, sizeof(getdetails));
if (rc) {
put_msg (TRUE, MSG_QUERY_CONFIG_ERROR, "nwlnkipx");
put_msg (TRUE, get_emsg(rc));
goto errorexit;
}
printf("\n");
if (getdetails.NetworkNumber != 0) {
sprintf (network, "%.8x", REORDER_ULONG(getdetails.NetworkNumber));
put_msg (FALSE, MSG_SHOW_INTERNAL_NET, network);
}
//
// The NicId 0 query returns the total number.
//
nicidcount = getdetails.NicId;
for (nicid = 1; nicid <= nicidcount; nicid++) {
getdetails.NicId = nicid;
rc = do_isnipxioctl(isnipxfd, MIPX_CONFIG, (char *)&getdetails, sizeof(getdetails));
if (rc) {
continue;
}
sprintf (nicidbuf, "%d", nicid);
sprintf (network, "%.8x", REORDER_ULONG(getdetails.NetworkNumber));
switch (getdetails.FrameType) {
case 0: frametype = load_msg (MSG_ETHERNET_II); break;
case 1: frametype = load_msg (MSG_802_3); break;
case 2: frametype = load_msg (MSG_802_2); break;
case 3: frametype = load_msg (MSG_SNAP); break;
case 4: frametype = load_msg (MSG_ARCNET); break;
default: frametype = load_msg (MSG_UNKNOWN); break;
}
sprintf (adaptername, "%ws", getdetails.AdapterName);
sprintf (node, "%2.2x%2.2x%2.2x%2.2x%2.2x%2.2x",
getdetails.Node[0],
getdetails.Node[1],
getdetails.Node[2],
getdetails.Node[3],
getdetails.Node[4],
getdetails.Node[5]);
special[1] = '\0';
if (getdetails.BindingSet) {
special[0] = '*';
showlegend |= 1;
} else if (getdetails.Type == 2) {
special[0] = '+';
showlegend |= 2;
} else if (getdetails.Type == 3) {
special[0] = '-';
showlegend |= 4;
} else {
special[0] = '\0';
}
put_msg (FALSE, MSG_SHOW_NET_NUMBER,
nicidbuf,
network,
frametype,
adaptername,
node,
special);
LocalFree (frametype);
}
if (showlegend) {
if (showlegend & 1) {
put_msg (FALSE, MSG_LEGEND_BINDING_SET);
}
if (showlegend & 2) {
put_msg (FALSE, MSG_LEGEND_ACTIVE_WAN);
}
if (showlegend & 4) {
put_msg (FALSE, MSG_LEGEND_DOWN_WAN);
}
printf("\n");
}
}
errorexit:
/** Close up and exit **/
if (isnipxfd != INVALID_HANDLE) NtClose(isnipxfd);
if (nwlinkfd != INVALID_HANDLE) NtClose(nwlinkfd);
if (isnripfd != INVALID_HANDLE) NtClose(isnripfd);
exit(0);
}
/*page***************************************************************
g e t _ e m s g
Get an error message for an error
Arguments - None
Returns - Does not return
********************************************************************/
unsigned long get_emsg(int rc)
{
/**
We have 3 defined error codes that can come back.
1 - EINVAL means that we sent down parameters wrong
(SHOULD NEVER HAPPEN)
2 - ERANGE means that the board number is invalid
(CAN HAPPEN IF USER ENTERS BAD BOARD)
3 - ENOENT means that on remove - the address given
is not in the source routing table.
**/
switch (rc) {
case EINVAL:
return MSG_INTERNAL_ERROR;
case ERANGE:
return MSG_INVALID_BOARD;
case ENOENT:
return MSG_ADDRESS_NOT_FOUND;
default:
return MSG_UNKNOWN_ERROR;
}
}
/*page***************************************************************
d o _ i s n i p x i o c t l
Do the equivalent of a stream ioctl to isnipx
Arguments - fd = Handle to put on
cmd = Command to send
datap = Ptr to ctrl buffer
dlen = Ptr to len of data buffer
Returns - 0 = OK
else = Error
********************************************************************/
int do_isnipxioctl(HANDLE fd, int cmd, char *datap, int dlen)
{
NTSTATUS Status;
UCHAR buffer[sizeof(NWLINK_ACTION) + sizeof(ISN_ACTION_GET_DETAILS) - 1];
PNWLINK_ACTION action;
IO_STATUS_BLOCK IoStatusBlock;
int rc;
/** Fill out the structure **/
action = (PNWLINK_ACTION)buffer;
action->Header.TransportId = ISN_ACTION_TRANSPORT_ID;
action->OptionType = NWLINK_OPTION_CONTROL;
action->BufferLength = sizeof(ULONG) + dlen;
action->Option = cmd;
RtlMoveMemory(action->Data, datap, dlen);
/** Issue the ioctl **/
Status = NtDeviceIoControlFile(
fd,
NULL,
NULL,
NULL,
&IoStatusBlock,
IOCTL_TDI_ACTION,
NULL,
0,
action,
FIELD_OFFSET(NWLINK_ACTION,Data) + dlen);
if (Status != STATUS_SUCCESS) {
if (Status == STATUS_INVALID_PARAMETER) {
rc = ERANGE;
} else {
rc = EINVAL;
}
} else {
if (dlen > 0) {
RtlMoveMemory (datap, action->Data, dlen);
}
rc = 0;
}
return rc;
}
//*****************************************************************************
//
// Name: put_msg
//
// Description: Reads a message resource, formats it in the current language
// and displays the message.
//
// NOTE: This routine was stolen from net\sockets\tcpcmd\common2\util.c.
//
// Parameters: error - TRUE if this is an error message.
// unsigned long MsgNum: ID of the message resource.
//
// Returns: unsigned long: number of characters displayed.
//
// History:
// 01/05/93 JayPh Created.
//
//*****************************************************************************
unsigned long put_msg(BOOLEAN error, unsigned long MsgNum, ... )
{
unsigned long msglen;
char *vp;
va_list arglist;
va_start( arglist, MsgNum );
msglen = FormatMessage( FORMAT_MESSAGE_ALLOCATE_BUFFER |
FORMAT_MESSAGE_FROM_HMODULE,
NULL,
MsgNum,
0L, // Default country ID.
(LPTSTR)&vp,
0,
&arglist );
if ( msglen == 0 )
{
return ( 0 );
}
fprintf( error ? stderr : stdout, "%s", vp );
LocalFree( vp );
return ( msglen );
}
//*****************************************************************************
//
// Name: load_msg
//
// Description: Reads and formats a message resource and returns a pointer
// to the buffer containing the formatted message. It is the
// responsibility of the caller to free the buffer.
//
// NOTE: This routine was stolen from net\sockets\tcpcmd\common2\util.c.
//
// Parameters: unsigned long MsgNum: ID of the message resource.
//
// Returns: char *: pointer to the message buffer, NULL if error.
//
// History:
// 01/05/93 JayPh Created.
//
//*****************************************************************************
char *load_msg( unsigned long MsgNum, ... )
{
unsigned long msglen;
char *vp;
va_list arglist;
va_start( arglist, MsgNum );
msglen = FormatMessage( FORMAT_MESSAGE_ALLOCATE_BUFFER |
FORMAT_MESSAGE_FROM_HMODULE,
NULL,
MsgNum,
0L, // Default country ID.
(LPTSTR)&vp,
0,
&arglist );
if ( msglen == 0 )
{
return(0);
}
return ( vp );
}
#define MAX_NETWORK_INTERFACES 255
typedef struct router_info
{
ULONG NetNumber;
USHORT TickCount;
USHORT HopCount;
USHORT NicId;
UCHAR InterfaceNumber[10];
} ROUTER_INFO, *PROUTER_INFO;
/*page***************************************************************
s h o w _ r o u t e r _ t a b l e
Display the IPX routing table
Arguments - FileHandle = Router File Handle
IoStatusBlock = Device IO Status Block
Returns - Does not return
********************************************************************/
VOID
show_router_table(
PHANDLE FileHandle,
PIO_STATUS_BLOCK IoStatusBlock
)
{
SHOW_NIC_INFO nis[MAX_NETWORK_INTERFACES];
ULONG NetNumber;
char InterfaceNumber[10];
NTSTATUS Status;
USHORT index, i, NumEntries, count;
char router_entry[128];
char buffer[32];
PROUTER_INFO RouterInfo = NULL;
if (*FileHandle == INVALID_HANDLE) {
put_msg(TRUE, MSG_IPXROUTER_NOT_STARTED );
goto exit_show_table;
}
/** First get the Network numbers for all interfaces **/
index = 0;
while(TRUE) {
Status = NtDeviceIoControlFile(
*FileHandle, // HANDLE to File
NULL, // HANDLE to Event
NULL, // ApcRoutine
NULL, // ApcContext
IoStatusBlock, // IO_STATUS_BLOCK
IOCTL_IPXROUTER_SHOWNICINFO, // IoControlCode
&index, // Input Buffer
sizeof(USHORT), // Input Buffer Length
&nis[index], // Output Buffer
sizeof(SHOW_NIC_INFO)); // Output Buffer Length
if(IoStatusBlock->Status == STATUS_NO_MORE_ENTRIES) {
break;
}
index ++;
if(Status != STATUS_SUCCESS) {
sprintf(buffer, "%x", Status);
put_msg(TRUE, MSG_SHOWSTATS_FAILED, buffer);
goto exit_show_table;
}
if (index >= MAX_NETWORK_INTERFACES) {
// break out of this loop if there are more than 255 network
// interfaces because we only have storage for 255.
break;
}
}
Status = NtDeviceIoControlFile(
*FileHandle, // HANDLE to File
NULL, // HANDLE to Event
NULL, // ApcRoutine
NULL, // ApcContext
IoStatusBlock, // IO_STATUS_BLOCK
IOCTL_IPXROUTER_SNAPROUTES, // IoControlCode
NULL, // Input Buffer
0, // Input Buffer Length
NULL, // Output Buffer
0); // Output Buffer Length
if (IoStatusBlock->Status != STATUS_SUCCESS) {
sprintf(buffer, "%x", Status);
put_msg(TRUE, MSG_SNAPROUTES_FAILED, buffer);
goto exit_show_table;
}
// first determine the number of router table entries to
// allocate sufficient storage
NumEntries = 0;
while(TRUE) {
Status = NtDeviceIoControlFile(
*FileHandle, // HANDLE to File
NULL, // HANDLE to Event
NULL, // ApcRoutine
NULL, // ApcContext
IoStatusBlock, // IO_STATUS_BLOCK
IOCTL_IPXROUTER_GETNEXTROUTE, // IoControlCode
NULL, // Input Buffer
0, // Input Buffer Length
&rte, // Output Buffer
sizeof(IPX_ROUTE_ENTRY)); // Output Buffer Length
if(IoStatusBlock->Status == STATUS_NO_MORE_ENTRIES) {
break;
}
if(Status != STATUS_SUCCESS) {
sprintf(buffer,"%x",Status);
put_msg(TRUE, MSG_GETNEXTROUTE_FAILED, buffer);
goto exit_show_table;
}
NumEntries ++;
}
RouterInfo = (PROUTER_INFO) LocalAlloc(LPTR, sizeof(ROUTER_INFO) * NumEntries);
if(!RouterInfo) {
put_msg(FALSE, MSG_INSUFFICIENT_MEMORY);
goto exit_show_table;
}
Status = NtDeviceIoControlFile(
*FileHandle, // HANDLE to File
NULL, // HANDLE to Event
NULL, // ApcRoutine
NULL, // ApcContext
IoStatusBlock, // IO_STATUS_BLOCK
IOCTL_IPXROUTER_SNAPROUTES, // IoControlCode
NULL, // Input Buffer
0, // Input Buffer Length
NULL, // Output Buffer
0); // Output Buffer Length
if (IoStatusBlock->Status != STATUS_SUCCESS) {
sprintf(buffer, "%x", Status);
put_msg(TRUE, MSG_SNAPROUTES_FAILED, buffer);
goto exit_show_table;
}
index = 0;
while(TRUE) {
Status = NtDeviceIoControlFile(
*FileHandle, // HANDLE to File
NULL, // HANDLE to Event
NULL, // ApcRoutine
NULL, // ApcContext
IoStatusBlock, // IO_STATUS_BLOCK
IOCTL_IPXROUTER_GETNEXTROUTE, // IoControlCode
NULL, // Input Buffer
0, // Input Buffer Length
&rte, // Output Buffer
sizeof(IPX_ROUTE_ENTRY)); // Output Buffer Length
if(IoStatusBlock->Status == STATUS_NO_MORE_ENTRIES) {
break;
}
if(Status != STATUS_SUCCESS) {
sprintf(buffer,"%x",Status);
put_msg(TRUE, MSG_GETNEXTROUTE_FAILED, buffer);
goto exit_show_table;
}
// make sure we don't exceed the number of entries
if (index > NumEntries) {
break;
}
// get net nr in "on the wire" order
GETLONG2ULONG(&(RouterInfo[index].NetNumber), rte.Network);
// find out the matching Network number based on NIC ID
for(i=0; i < MAX_NETWORK_INTERFACES; i++) {
if(rte.NicId == nis[i].NicId) {
sprintf(RouterInfo[index].InterfaceNumber, "%.2x%.2x%.2x%.2x",
nis[i].Network[0],
nis[i].Network[1],
nis[i].Network[2],
nis[i].Network[3]);
break;
}
}
RouterInfo[index].TickCount = rte.TickCount;
RouterInfo[index].HopCount = rte.HopCount;
RouterInfo[index].NicId = rte.NicId;
index++;
}
// Now sort the entries by net number
qsort( (void*) RouterInfo,
NumEntries,
sizeof(ROUTER_INFO),
(PQSORT_COMPARE)CompareNetNumber );
put_msg(FALSE, MSG_ROUTER_TABLE_HEADER);
for(index =0, count = 0; index < NumEntries; index++, count++)
{
if (count > 50) {
count = 0;
// display router table header every 25 entries
// to make reading the table easier.
put_msg(FALSE, MSG_ROUTER_TABLE_HEADER);
}
printf("%.8x %6d %2d %-16s %d\n",
RouterInfo[index].NetNumber,
RouterInfo[index].TickCount,
RouterInfo[index].HopCount,
RouterInfo[index].InterfaceNumber,
RouterInfo[index].NicId );
}
/** Close up and exit **/
exit_show_table:
if (RouterInfo) LocalFree(RouterInfo);
if (isnipxfd != INVALID_HANDLE) NtClose(isnipxfd);
if (nwlinkfd != INVALID_HANDLE) NtClose(nwlinkfd);
if (isnripfd != INVALID_HANDLE) NtClose(isnripfd);
exit(0);
}
int _CRTAPI1 CompareNetNumber( void * p0, void * p1)
{
PROUTER_INFO pLeft = (PROUTER_INFO) p0;
PROUTER_INFO pRight = (PROUTER_INFO) p1;
if(pLeft->NetNumber == pRight->NetNumber)
return(0);
if(pLeft->NetNumber > pRight->NetNumber)
return(1);
else
return(-1);
}
PUCHAR DeviceType[2] = { "LAN", "WAN" };
PUCHAR NicState[4] = { "CLOSED", "CLOSING", "ACTIVE", "PENDING_OPEN" };
/*page***************************************************************
s h o w _ s t a t s
Displays IPX internal routing statistics
Arguments - FileHandle = Router File Handle
IoStatusBlock = Device IO Status Block
Returns - Does not return
********************************************************************/
VOID
show_stats(
PHANDLE FileHandle,
PIO_STATUS_BLOCK IoStatusBlock
)
{
SHOW_NIC_INFO nis;
USHORT index, i;
char NicId[4];
char NetworkNumber[10];
char RipRcvd[32], RipSent[32];
char RoutedRcvd[32], RoutedSent[32];
char Type20Rcvd[32], Type20Sent[32];
char BadRcvd[32];
char buffer[32];
NTSTATUS Status;
if (*FileHandle == INVALID_HANDLE) {
put_msg(TRUE, MSG_IPXROUTER_NOT_STARTED );
goto end_stats;
}
index = 0;
while(TRUE) {
Status = NtDeviceIoControlFile(
*FileHandle, // HANDLE to File
NULL, // HANDLE to Event
NULL, // ApcRoutine
NULL, // ApcContext
IoStatusBlock, // IO_STATUS_BLOCK
IOCTL_IPXROUTER_SHOWNICINFO, // IoControlCode
&index, // Input Buffer
sizeof(USHORT), // Input Buffer Length
&nis, // Output Buffer
sizeof(nis)); // Output Buffer Length
if(IoStatusBlock->Status == STATUS_NO_MORE_ENTRIES) {
goto end_stats;
}
index ++;
if(Status != STATUS_SUCCESS) {
sprintf(buffer, "%x", Status);
put_msg(TRUE, MSG_SHOWSTATS_FAILED, buffer);
goto end_stats;
}
sprintf(NicId, "%d", nis.NicId);
sprintf(NetworkNumber,
"%.2x%.2x%.2x%.2x",
nis.Network[0],
nis.Network[1],
nis.Network[2],
nis.Network[3]);
sprintf(RipRcvd, "%-8d", nis.StatRipReceived);
sprintf(RipSent, "%-8d", nis.StatRipSent);
sprintf(RoutedRcvd, "%-8d", nis.StatRoutedReceived);
sprintf(RoutedSent, "%-8d", nis.StatRoutedSent);
sprintf(Type20Rcvd, "%-8d", nis.StatType20Received);
sprintf(Type20Sent, "%-8d", nis.StatType20Sent);
sprintf(BadRcvd, "%-8d", nis.StatBadReceived);
put_msg(FALSE,
MSG_SHOW_STATISTICS,
NicId,
NetworkNumber,
RipRcvd,
RipSent,
Type20Rcvd,
Type20Sent,
RoutedRcvd,
RoutedSent,
BadRcvd);
}
/** Close up and exit **/
end_stats:
if (isnipxfd != INVALID_HANDLE) NtClose(isnipxfd);
if (nwlinkfd != INVALID_HANDLE) NtClose(nwlinkfd);
if (isnripfd != INVALID_HANDLE) NtClose(isnripfd);
exit(0);
}
/*page***************************************************************
c l e a r _ s t a t s
Clears the IPX internal routing statistics
Arguments - FileHandle = Router File Handle
IoStatusBlock = Device IO Status Block
Returns - Does not return
********************************************************************/
VOID
clear_stats(
PHANDLE FileHandle,
PIO_STATUS_BLOCK IoStatusBlock
)
{
NTSTATUS Status;
char buffer[32];
if (*FileHandle == INVALID_HANDLE) {
put_msg(TRUE, MSG_IPXROUTER_NOT_STARTED );
goto end_clearstats;
}
Status = NtDeviceIoControlFile(
*FileHandle, // HANDLE to File
NULL, // HANDLE to Event
NULL, // ApcRoutine
NULL, // ApcContext
IoStatusBlock, // IO_STATUS_BLOCK
IOCTL_IPXROUTER_ZERONICSTATISTICS, // IoControlCode
NULL, // Input Buffer
0, // Input Buffer Length
NULL, // Output Buffer
0); // Output Buffer Length
if(Status != STATUS_SUCCESS) {
sprintf(buffer, "%x", Status);
put_msg(TRUE, MSG_CLEAR_STATS_FAILED, buffer);
}
/** Close up and exit **/
end_clearstats:
if (isnipxfd != INVALID_HANDLE) NtClose(isnipxfd);
if (nwlinkfd != INVALID_HANDLE) NtClose(nwlinkfd);
if (isnripfd != INVALID_HANDLE) NtClose(isnripfd);
exit(0);
}
typedef struct server_info
{
USHORT ObjectType;
UCHAR ObjectName[100];
UCHAR IpxAddress[12];
} SERVER_INFO, *PSERVER_INFO;
/*page***************************************************************
s h o w _ s e r v e r s
Display the servers from the SAP table
Arguments - servertype = Type of servers to display
Defaults to show all server types
Returns - Does not return
********************************************************************/
VOID
show_servers(int servertype)
{
INT rc;
ULONG ObjectID = 0xFFFFFFFF;
UCHAR ObjectName[100];
USHORT ObjectType;
USHORT ScanType = servertype;
UCHAR IpxAddress[12];
USHORT i;
USHORT index, count, NumServers;
PSERVER_INFO ServerInfo = NULL;
if(rc = SapLibInit() != SAPRETURN_SUCCESS) {
put_msg(TRUE, MSG_SAP_NOT_STARTED);
goto show_servers_end;
}
memset(&ObjectName, 0, 100);
// find out how many servers are there so that we can allocate
// sufficient storage
NumServers = 0;
while((rc = SapScanObject(&ObjectID,
ObjectName,
&ObjectType,
ScanType)) == SAPRETURN_SUCCESS)
{
NumServers++;
}
ServerInfo = (PSERVER_INFO) LocalAlloc(LPTR, sizeof(SERVER_INFO) * NumServers);
if(!ServerInfo)
{
put_msg(FALSE, MSG_INSUFFICIENT_MEMORY);
goto show_servers_end;
}
index = 0;
ObjectID = 0xFFFFFFFF;
while((rc = SapScanObject(&ObjectID,
ObjectName,
&ObjectType,
ScanType)) == SAPRETURN_SUCCESS)
{
if (index > NumServers) {
break;
}
// get object address
SapGetObjectName(ObjectID,
ObjectName,
&ObjectType,
IpxAddress);
ServerInfo[index].ObjectType = ObjectType;
strcpy(ServerInfo[index].ObjectName, ObjectName);
CopyMemory(ServerInfo[index].IpxAddress, IpxAddress, 12);
index++;
}
// Now sort the entries by server name
qsort( (void*) ServerInfo,
NumServers,
sizeof(SERVER_INFO),
(PQSORT_COMPARE)CompareServerNames );
if(servertype == SHOW_ALL_SERVERS)
put_msg(FALSE, MSG_SHOW_ALL_SERVERS_HEADER);
else
put_msg(FALSE, MSG_SHOW_SPECIFIC_SERVER_HEADER);
for(index = 0, count = 0; index < NumServers; index++, count++)
{
if (count > 50) {
// write the table header for every 50 entries
// to make this more readable.
count = 0;
if(servertype == SHOW_ALL_SERVERS)
put_msg(FALSE, MSG_SHOW_ALL_SERVERS_HEADER);
else
put_msg(FALSE, MSG_SHOW_SPECIFIC_SERVER_HEADER);
}
for(i=0; i<4; i++) {
printf("%.2x", ServerInfo[index].IpxAddress[i]);
}
printf(".");
for(i=4; i<10; i++) {
printf("%.2x", ServerInfo[index].IpxAddress[i]);
}
if(servertype == SHOW_ALL_SERVERS) {
printf(" %-6d", ServerInfo[index].ObjectType);
}
printf(" %s\n", ServerInfo[index].ObjectName);
}
/** Close up and exit **/
show_servers_end:
if (ServerInfo) LocalFree(ServerInfo);
if (isnipxfd != INVALID_HANDLE) NtClose(isnipxfd);
if (nwlinkfd != INVALID_HANDLE) NtClose(nwlinkfd);
if (isnripfd != INVALID_HANDLE) NtClose(isnripfd);
exit(0);
}
int _CRTAPI1 CompareServerNames( void * p0, void * p1)
{
PSERVER_INFO pLeft = (PSERVER_INFO) p0;
PSERVER_INFO pRight = (PSERVER_INFO) p1;
return(strcmp(pLeft->ObjectName, pRight->ObjectName));
}