89e4517
--- net-tools-1.60/statistics.c.sctp	2006-04-13 10:06:45.000000000 -0400
89e4517
+++ net-tools-1.60/statistics.c	2006-04-13 10:06:45.000000000 -0400
89e4517
@@ -20,7 +20,7 @@
89e4517
 #define UFWARN(x)
89e4517
 #endif
89e4517
 
89e4517
-int print_static,f_raw,f_tcp,f_udp,f_unknown = 1;
89e4517
+int print_static,f_raw,f_tcp,f_udp,f_sctp,f_unknown = 1;
89e4517
 
89e4517
 enum State {
89e4517
     number = 0, opt_number, i_forward, i_inp_icmp, i_outp_icmp, i_rto_alg,
89e4517
@@ -225,6 +225,27 @@
89e4517
      { "TCPLoss", N_("%u TCP data loss events") },
89e4517
 };
89e4517
 
89e4517
+struct entry Sctptab[] =
89e4517
+{
89e4517
+    {"SctpCurrEstab", N_("%u Current Associations"), number},
89e4517
+    {"SctpActiveEstabs", N_("%u Active Associations"), number},
89e4517
+    {"SctpPassiveEstabs", N_("%u Passive Associations"), number},
89e4517
+    {"SctpAborteds", N_("%u Number of Aborteds "), number},
89e4517
+    {"SctpShutdowns", N_("%u Number of Graceful Terminations"), number},
89e4517
+    {"SctpOutOfBlues", N_("%u Number of Out of Blue packets"), number},
89e4517
+    {"SctpChecksumErrors", N_("%u Number of Packets with invalid Checksum"), number},
89e4517
+    {"SctpOutCtrlChunks", N_("%u Number of control chunks sent"), number},
89e4517
+    {"SctpOutOrderChunks", N_("%u Number of ordered chunks sent"), number},
89e4517
+    {"SctpOutUnorderChunks", N_("%u Number of Unordered chunks sent"), number},
89e4517
+    {"SctpInCtrlChunks", N_("%u Number of control chunks received"), number},
89e4517
+    {"SctpInOrderChunks", N_("%u Number of ordered chunks received"), number},
89e4517
+    {"SctpInUnorderChunks", N_("%u Number of Unordered chunks received"), number},
89e4517
+    {"SctpFragUsrMsgs", N_("%u Number of messages fragmented"), number},
89e4517
+    {"SctpReasmUsrMsgs", N_("%u Number of messages reassembled "), number},
89e4517
+    {"SctpOutSCTPPacks", N_("%u Number of SCTP packets sent"), number},
89e4517
+    {"SctpInSCTPPacks", N_("%u Number of SCTP packets received"), number},
89e4517
+};
89e4517
+
89e4517
 struct tabtab {
89e4517
     char *title;
89e4517
     struct entry *tab;
89e4517
@@ -238,6 +259,7 @@
89e4517
     {"Icmp", Icmptab, sizeof(Icmptab), &f_raw},
89e4517
     {"Tcp", Tcptab, sizeof(Tcptab), &f_tcp},
89e4517
     {"Udp", Udptab, sizeof(Udptab), &f_udp},
89e4517
+    {"Sctp", Sctptab, sizeof(Sctptab), &f_sctp},
89e4517
     {"TcpExt", Tcpexttab, sizeof(Tcpexttab), &f_tcp},
89e4517
     {NULL}
89e4517
 };
89e4517
@@ -385,12 +407,39 @@
89e4517
   return;
89e4517
 }
89e4517
 
89e4517
+/* Process a file with name-value lines (like /proc/net/sctp/snmp) */
89e4517
+void process_fd2(FILE *f, const char *filename)
89e4517
+{
89e4517
+    char buf1[1024];
89e4517
+    char *sp;
89e4517
+    struct tabtab *tab;
89e4517
+    
89e4517
+    tab = newtable(snmptabs, "Sctp");
89e4517
+    
89e4517
+    while (fgets(buf1, sizeof buf1, f)) {
89e4517
+	sp = buf1 + strcspn(buf1, " \t\n");
89e4517
+	if (!sp)
89e4517
+	    goto formaterr;
89e4517
+	*sp = '\0';
89e4517
+    	sp++;
89e4517
+
89e4517
+	sp += strspn(sp, " \t\n"); 
89e4517
 
89e4517
-int parsesnmp(int flag_raw, int flag_tcp, int flag_udp)
89e4517
+	if (*sp != '\0' && *(tab->flag)) 	
89e4517
+	    printval(tab, buf1, strtoul(sp, 0, 10));
89e4517
+    }
89e4517
+  return;
89e4517
+  
89e4517
+formaterr:
89e4517
+  fprintf(stderr,_("error parsing %s\n"), filename);
89e4517
+  return;
89e4517
+}
89e4517
+
89e4517
+int parsesnmp(int flag_raw, int flag_tcp, int flag_udp, int flag_sctp)
89e4517
 {
89e4517
     FILE *f;
89e4517
 
89e4517
-    f_raw = flag_raw; f_tcp = flag_tcp; f_udp = flag_udp;
89e4517
+    f_raw = flag_raw; f_tcp = flag_tcp; f_udp = flag_udp; f_sctp = flag_sctp;
89e4517
 
89e4517
     f = fopen("/proc/net/snmp", "r");
89e4517
     if (!f) {
89e4517
@@ -418,6 +467,16 @@
89e4517
     
89e4517
         fclose(f);
89e4517
     }
89e4517
+
89e4517
+    f = fopen("/proc/net/sctp/snmp", "r");
89e4517
+    if (f) {
89e4517
+	process_fd2(f,"/proc/net/sctp/snmp");
89e4517
+	if (ferror(f))
89e4517
+	    perror("/proc/net/sctp/snmp");
89e4517
+
89e4517
+	fclose(f);
89e4517
+    }
89e4517
+
89e4517
     return(0);
89e4517
 }
89e4517
     
89e4517
--- net-tools-1.60/netstat.c.sctp	2006-04-13 10:06:45.000000000 -0400
89e4517
+++ net-tools-1.60/netstat.c	2006-04-13 10:10:23.000000000 -0400
89e4517
@@ -58,6 +58,7 @@
89e4517
  *
89e4517
  *990420 {1.38} Tuan Hoang              removed a useless assignment from igmp_do_one()
89e4517
  *20010404 {1.39} Arnaldo Carvalho de Melo - use setlocale
89e4517
+ *20050516 {1.40} Ivan Skytte Joergensen:Added SCTP support
89e4517
  *
89e4517
  *              This program is free software; you can redistribute it
89e4517
  *              and/or  modify it under  the terms of  the GNU General
89e4517
@@ -108,7 +109,7 @@
89e4517
 #endif
89e4517
 
89e4517
 /* prototypes for statistics.c */
89e4517
-int parsesnmp(int, int, int);
89e4517
+int parsesnmp(int, int, int, int);
89e4517
 void inittab(void);
89e4517
 
89e4517
 typedef enum {
89e4517
@@ -119,6 +120,29 @@
89e4517
     SS_DISCONNECTING		/* in process of disconnecting  */
89e4517
 } socket_state;
89e4517
 
89e4517
+
89e4517
+#define SCTP_NSTATES  9         /* The number of states in array*/
89e4517
+
89e4517
+static const char *sctp_state[] = {
89e4517
+    N_("EMPTY"),
89e4517
+    N_("CLOSED"),
89e4517
+    N_("COOKIE_WAIT"),
89e4517
+    N_("COOKIE_ECHOED"),
89e4517
+    N_("ESTABLISHED"),
89e4517
+    N_("SHUTDOWN_PENDING"),
89e4517
+    N_("SHUTDOWN_SENT"),
89e4517
+    N_("SHUTDOWN_RECEIVED"),
89e4517
+    N_("SHUTDOWN_ACK_SENT")
89e4517
+};
89e4517
+
89e4517
+#define SCTP_NTYPES 3           /* The number of types in array */
89e4517
+
89e4517
+static const char *sctp_type[] = {
89e4517
+    N_("udp"),
89e4517
+    N_("udp-high-bw"),
89e4517
+    N_("tcp")
89e4517
+};
89e4517
+
89e4517
 #define SO_ACCEPTCON    (1<<16)	/* performed a listen           */
89e4517
 #define SO_WAITDATA     (1<<17)	/* wait data to read            */
89e4517
 #define SO_NOSPACE      (1<<18)	/* no space to write            */
89e4517
@@ -150,6 +174,7 @@
89e4517
 int flag_raw = 0;
89e4517
 int flag_tcp = 0;
89e4517
 int flag_udp = 0;
89e4517
+int flag_sctp= 0;
89e4517
 int flag_igmp= 0;
89e4517
 int flag_rom = 0;
89e4517
 int flag_exp = 1;
89e4517
@@ -1189,6 +1214,365 @@
89e4517
 	       udp_do_one);
89e4517
 }
89e4517
 
89e4517
+static const char *sctp_socket_type_str(int type) {
89e4517
+    if(type>=0 && type
89e4517
+	return sctp_type[type];
89e4517
+    else {
89e4517
+	static char type_str_buf[64];
89e4517
+	sprintf(type_str_buf,"UNKNOWN(%d)",type);
89e4517
+	return type_str_buf;
89e4517
+    }
89e4517
+}
89e4517
+
89e4517
+static const char *sctp_state_str(int state)
89e4517
+{
89e4517
+    if(state>=0 && state
89e4517
+	return sctp_state[state];
89e4517
+    else {
89e4517
+	static char state_str_buf[64];
89e4517
+	sprintf(state_str_buf,"UNKNOWN(%d)",state);
89e4517
+	return state_str_buf;
89e4517
+    }
89e4517
+}
89e4517
+
89e4517
+static const char *sctp_socket_state_str(int state)
89e4517
+{
89e4517
+    if(state>=0 && state<=10)
89e4517
+        return tcp_state[state];
89e4517
+    else {
89e4517
+	static char state_str_buf[64];
89e4517
+	sprintf(state_str_buf,"UNKNOWN(%d)",state);
89e4517
+	return state_str_buf;
89e4517
+    }
89e4517
+}
89e4517
+
89e4517
+static struct aftype *process_sctp_addr_str(const char *addr_str, struct sockaddr *sa)
89e4517
+{
89e4517
+    if (strchr(addr_str,':')) {
89e4517
+#if HAVE_AFINET6
89e4517
+	extern struct aftype inet6_aftype;
89e4517
+	/* Demangle what the kernel gives us */
89e4517
+	struct in6_addr in6;
89e4517
+	char addr6_str[INET6_ADDRSTRLEN];
89e4517
+	unsigned u0,u1,u2,u3,u4,u5,u6,u7;
89e4517
+	sscanf(addr_str, "%04X:%04X:%04X:%04X:%04X:%04X:%04X:%04X",
89e4517
+	       &u0, &u1, &u2, &u3, &u4, &u5, &u6, &u7;;
89e4517
+	in6.s6_addr16[0] = htons(u0);
89e4517
+	in6.s6_addr16[1] = htons(u1);
89e4517
+	in6.s6_addr16[2] = htons(u2);
89e4517
+	in6.s6_addr16[3] = htons(u3);
89e4517
+	in6.s6_addr16[4] = htons(u4);
89e4517
+	in6.s6_addr16[5] = htons(u5);
89e4517
+	in6.s6_addr16[6] = htons(u6);
89e4517
+	in6.s6_addr16[7] = htons(u7);
89e4517
+
89e4517
+	inet_ntop(AF_INET6, &in6, addr6_str, sizeof(addr6_str));
89e4517
+	inet6_aftype.input(1, addr6_str, sa);
89e4517
+	sa->sa_family = AF_INET6;
89e4517
+#endif
89e4517
+    } else {
89e4517
+    	((struct sockaddr_in*)sa)->sin_addr.s_addr = inet_addr(addr_str);
89e4517
+	sa->sa_family = AF_INET;
89e4517
+    }
89e4517
+    return get_afntype(sa->sa_family);
89e4517
+}
89e4517
+
89e4517
+static void sctp_eps_do_one(int lnr, char *line)
89e4517
+{
89e4517
+    char buffer[1024];
89e4517
+    int type, state, port;
89e4517
+    int uid;
89e4517
+    unsigned long inode;
89e4517
+    
89e4517
+    struct aftype *ap;
89e4517
+#if HAVE_AFINET6
89e4517
+    struct sockaddr_in6 localaddr;
89e4517
+#else
89e4517
+    struct sockaddr_in localaddr;
89e4517
+#endif
89e4517
+    const char *sty_str;
89e4517
+    const char *sst_str;
89e4517
+    const char *lport_str;
89e4517
+    const char *uid_str;
89e4517
+    const char *inode_str;
89e4517
+    const char *pladdr_str;
89e4517
+    char *laddrs_str;
89e4517
+    
89e4517
+    if(lnr == 0) {
89e4517
+	/* ENDPT     SOCK   STY SST HBKT LPORT   uid inode pladdr LADDRS*/
89e4517
+	return;
89e4517
+    }
89e4517
+    
89e4517
+    strtok(line," \t\n"); /*skip ptr*/
89e4517
+    strtok(0," \t\n");    /*skip ptr*/
89e4517
+    sty_str = strtok(0," \t\n");
89e4517
+    sst_str = strtok(0," \t\n");
89e4517
+    strtok(0," \t\n"); /*skip hash bucket*/
89e4517
+    lport_str=strtok(0," \t\n");
89e4517
+    uid_str = strtok(0," \t\n");
89e4517
+    inode_str = strtok(0," \t\n");
89e4517
+    pladdr_str = strtok(0," \t\n");
89e4517
+    laddrs_str=strtok(0,"\t\n");
89e4517
+    
89e4517
+    type = atoi(sty_str);
89e4517
+    state = atoi(sst_str);
89e4517
+    port = atoi(lport_str);
89e4517
+    uid = atoi(uid_str);
89e4517
+    inode = strtoul(inode_str,0,0);
89e4517
+    
89e4517
+    if(flag_sctp<=1) {
89e4517
+	/* only print the primary address */
89e4517
+	char local_addr[64];
89e4517
+	char local_port[16];
89e4517
+	
89e4517
+	ap = process_sctp_addr_str(pladdr_str, (struct sockaddr*)&localaddr);
89e4517
+	if(ap)
89e4517
+	    safe_strncpy(local_addr,
89e4517
+	                 ap->sprint((struct sockaddr *) &localaddr, flag_not),
89e4517
+	                 sizeof(local_addr));
89e4517
+	else
89e4517
+	    sprintf(local_addr,_("unsupported address family %d"), ((struct sockaddr*)&localaddr)->sa_family);
89e4517
+	
89e4517
+	snprintf(local_port, sizeof(local_port), "%s",
89e4517
+	         get_sname(htons(port), "sctp",
89e4517
+	                   flag_not & FLAG_NUM_PORT));
89e4517
+	
89e4517
+	printf("sctp                ");
89e4517
+	sprintf(buffer,"%s:%s", local_addr, local_port);
89e4517
+	printf("%-47s", buffer);
89e4517
+    	printf(" %-12s", sctp_socket_state_str(state));
89e4517
+    } else {
89e4517
+    	/*print all addresses*/
89e4517
+	const char *this_local_addr;
89e4517
+	int first=1;
89e4517
+	char local_port[16];
89e4517
+	snprintf(local_port, sizeof(local_port), "%s",
89e4517
+		 get_sname(htons(port), "sctp",
89e4517
+			   flag_not & FLAG_NUM_PORT));
89e4517
+	for(this_local_addr=strtok(laddrs_str," \t\n");
89e4517
+	    this_local_addr;
89e4517
+	    this_local_addr=strtok(0," \t\n"))
89e4517
+	{
89e4517
+	    char local_addr[64];
89e4517
+	    ap = process_sctp_addr_str(this_local_addr, (struct sockaddr*)&localaddr);
89e4517
+	    if(ap)
89e4517
+		safe_strncpy(local_addr,
89e4517
+		             ap->sprint((struct sockaddr *) &localaddr, flag_not),
89e4517
+		             sizeof(local_addr));
89e4517
+	    else
89e4517
+		sprintf(local_addr,_("unsupported address family %d"), ((struct sockaddr*)&localaddr)->sa_family);
89e4517
+
89e4517
+	    if(!first) printf("\n");
89e4517
+	    if(first)
89e4517
+	        printf("sctp                ");
89e4517
+	    else
89e4517
+	        printf("                    ");
89e4517
+	    sprintf(buffer,"%s:%s", local_addr, local_port);
89e4517
+	    printf("%-47s", buffer);
89e4517
+	    printf(" %-12s", first?sctp_socket_state_str(state):"");
89e4517
+	    first = 0;
89e4517
+	}
89e4517
+    }
89e4517
+
89e4517
+    finish_this_one(uid,inode,"");
89e4517
+}
89e4517
+
89e4517
+static void sctp_assoc_do_one(int lnr, char *line)
89e4517
+{
89e4517
+    char buffer[1024];
89e4517
+    int type, state, state2, lport,rport;
89e4517
+    int uid;
89e4517
+    unsigned rxqueue,txqueue;
89e4517
+    unsigned long inode;
89e4517
+    
89e4517
+    struct aftype *ap;
89e4517
+#if HAVE_AFINET6
89e4517
+    struct sockaddr_in6 localaddr,remoteaddr;
89e4517
+#else
89e4517
+    struct sockaddr_in localaddr,remoteaddr;
89e4517
+#endif
89e4517
+    const char *sty_str;
89e4517
+    const char *sst_str;
89e4517
+    const char *st_str;
89e4517
+    const char *txqueue_str;
89e4517
+    const char *rxqueue_str;
89e4517
+    const char *lport_str,*rport_str;
89e4517
+    const char *uid_str;
89e4517
+    const char *inode_str;
89e4517
+    const char *pladdr_str;
89e4517
+    char *laddrs_str;
89e4517
+    const char *praddr_str;
89e4517
+    char *raddrs_str;
89e4517
+    
89e4517
+    if(lnr == 0) {
89e4517
+	/* ASSOC     SOCK   STY SST ST HBKT tx_queue rx_queue uid inode LPORT RPORT pladdr praddr LADDRS <-> RADDRS*/
89e4517
+	return;
89e4517
+    }
89e4517
+    
89e4517
+    strtok(line," \t\n"); /*skip ptr*/
89e4517
+    strtok(0," \t\n");    /*skip ptr*/
89e4517
+    sty_str = strtok(0," \t\n");
89e4517
+    sst_str = strtok(0," \t\n");
89e4517
+    st_str = strtok(0," \t\n");
89e4517
+    strtok(0," \t\n"); /*skip hash bucket*/
89e4517
+    txqueue_str =  strtok(0," \t\n");
89e4517
+    rxqueue_str =  strtok(0," \t\n");
89e4517
+    uid_str = strtok(0," \t\n");
89e4517
+    inode_str = strtok(0," \t\n");
89e4517
+    lport_str=strtok(0," \t\n");
89e4517
+    rport_str=strtok(0," \t\n");
89e4517
+    pladdr_str = strtok(0," \t\n");
89e4517
+    praddr_str = strtok(0," \t\n");
89e4517
+    laddrs_str=strtok(0,"<->\t\n");
89e4517
+    raddrs_str=strtok(0,"<->\t\n");
89e4517
+
89e4517
+    type = atoi(sty_str);
89e4517
+    state = atoi(sst_str);
89e4517
+    state2 = atoi(st_str);
89e4517
+    txqueue = atoi(txqueue_str);
89e4517
+    rxqueue = atoi(rxqueue_str);
89e4517
+    uid = atoi(uid_str);
89e4517
+    inode = strtoul(inode_str,0,0);
89e4517
+    lport = atoi(lport_str);
89e4517
+    rport = atoi(rport_str);
89e4517
+    
89e4517
+    if(flag_sctp<=1) {
89e4517
+	/* only print the primary addresses */
89e4517
+	char local_addr[64];
89e4517
+	char local_port[16];
89e4517
+	char remote_addr[64];
89e4517
+	char remote_port[16];
89e4517
+	
89e4517
+	ap = process_sctp_addr_str(pladdr_str, (struct sockaddr*)&localaddr);
89e4517
+	if(ap)
89e4517
+	    safe_strncpy(local_addr,
89e4517
+	                 ap->sprint((struct sockaddr *) &localaddr, flag_not),
89e4517
+	                 sizeof(local_addr));
89e4517
+	else
89e4517
+	    sprintf(local_addr,_("unsupported address family %d"), ((struct sockaddr*)&localaddr)->sa_family);
89e4517
+	
89e4517
+	snprintf(local_port, sizeof(local_port), "%s",
89e4517
+	         get_sname(htons(lport), "sctp",
89e4517
+	                   flag_not & FLAG_NUM_PORT));
89e4517
+	
89e4517
+	ap = process_sctp_addr_str(praddr_str, (struct sockaddr*)&remoteaddr);
89e4517
+	if(ap)
89e4517
+	    safe_strncpy(remote_addr,
89e4517
+	                 ap->sprint((struct sockaddr *) &remoteaddr, flag_not),
89e4517
+	                 sizeof(remote_addr));
89e4517
+	else
89e4517
+	    sprintf(remote_addr,_("unsupported address family %d"), ((struct sockaddr*)&remoteaddr)->sa_family);
89e4517
+	
89e4517
+	snprintf(remote_port, sizeof(remote_port), "%s",
89e4517
+		 get_sname(htons(rport), "sctp",
89e4517
+			   flag_not & FLAG_NUM_PORT));
89e4517
+
89e4517
+	printf("sctp");
89e4517
+	printf("  %6u %6u ", rxqueue, txqueue);
89e4517
+	sprintf(buffer,"%s:%s", local_addr, local_port);
89e4517
+	printf("%-23s", buffer);
89e4517
+	printf(" ");
89e4517
+	sprintf(buffer,"%s:%s", remote_addr, remote_port);
89e4517
+	printf("%-23s", buffer);
89e4517
+    	printf(" %-12s", sctp_socket_state_str(state));
89e4517
+    } else {
89e4517
+    	/*print all addresses*/
89e4517
+	const char *this_local_addr;
89e4517
+	const char *this_remote_addr;
89e4517
+	char *ss1,*ss2;
89e4517
+	int first=1;
89e4517
+	char local_port[16];
89e4517
+	char remote_port[16];
89e4517
+	snprintf(local_port, sizeof(local_port), "%s",
89e4517
+	         get_sname(htons(lport), "sctp",
89e4517
+	                   flag_not & FLAG_NUM_PORT));
89e4517
+	snprintf(remote_port, sizeof(remote_port), "%s",
89e4517
+	         get_sname(htons(rport), "sctp",
89e4517
+	                   flag_not & FLAG_NUM_PORT));
89e4517
+
89e4517
+	this_local_addr=strtok_r(laddrs_str," \t\n",&ss1;;
89e4517
+	this_remote_addr=strtok_r(raddrs_str," \t\n",&ss2;;
89e4517
+	while(this_local_addr || this_remote_addr) {
89e4517
+	    char local_addr[64];
89e4517
+	    char remote_addr[64];
89e4517
+	    if(this_local_addr) {
89e4517
+		ap = process_sctp_addr_str(this_local_addr, (struct sockaddr*)&localaddr);
89e4517
+		if(ap)
89e4517
+		    safe_strncpy(local_addr,
89e4517
+		                 ap->sprint((struct sockaddr *) &localaddr, flag_not),
89e4517
+		                 sizeof(local_addr));
89e4517
+		else
89e4517
+		    sprintf(local_addr,_("unsupported address family %d"), ((struct sockaddr*)&localaddr)->sa_family);
89e4517
+	    }
89e4517
+	    if(this_remote_addr) {
89e4517
+		ap = process_sctp_addr_str(this_remote_addr, (struct sockaddr*)&remoteaddr);
89e4517
+		if(ap)
89e4517
+		    safe_strncpy(remote_addr,
89e4517
+		                 ap->sprint((struct sockaddr *) &remoteaddr, flag_not),
89e4517
+		                 sizeof(remote_addr));
89e4517
+		else
89e4517
+		    sprintf(remote_addr,_("unsupported address family %d"), ((struct sockaddr*)&remoteaddr)->sa_family);
89e4517
+	    }
89e4517
+
89e4517
+	    if(!first) printf("\n");
89e4517
+	    if(first)
89e4517
+		printf("sctp  %6u %6u ", rxqueue, txqueue);
89e4517
+	    else
89e4517
+		printf("                    ");
89e4517
+	    if(this_local_addr) {
89e4517
+		if(first)
89e4517
+		    sprintf(buffer,"%s:%s", local_addr, local_port);
89e4517
+		else
89e4517
+		    sprintf(buffer,"%s", local_addr);
89e4517
+		printf("%-23s", buffer);
89e4517
+	    } else
89e4517
+	    	printf("%-23s", "");
89e4517
+	    printf(" ");
89e4517
+	    if(this_remote_addr) {
89e4517
+		if(first)
89e4517
+		    sprintf(buffer,"%s:%s", remote_addr, remote_port);
89e4517
+		else
89e4517
+		    sprintf(buffer,"%s", remote_addr);
89e4517
+		printf("%-23s", buffer);
89e4517
+	    } else
89e4517
+		printf("%-23s", "");
89e4517
+
89e4517
+	    printf(" %-12s", first?sctp_socket_state_str(state):"");
89e4517
+
89e4517
+	    first = 0;
89e4517
+	    this_local_addr=strtok_r(0," \t\n",&ss1;;
89e4517
+	    this_remote_addr=strtok_r(0," \t\n",&ss2;;
89e4517
+	}
89e4517
+    }
89e4517
+
89e4517
+    finish_this_one(uid,inode,"");
89e4517
+}
89e4517
+
89e4517
+static int sctp_info_eps(void)
89e4517
+{
89e4517
+#if !defined(_PATH_PROCNET_SCTP_EPS)
89e4517
+#define	_PATH_PROCNET_SCTP_EPS	"/proc/net/sctp/eps"
89e4517
+#endif
89e4517
+    INFO_GUTS(_PATH_PROCNET_SCTP_EPS, "AF INET (sctp)",
89e4517
+              sctp_eps_do_one);
89e4517
+}
89e4517
+
89e4517
+static int sctp_info_assocs(void)
89e4517
+{
89e4517
+#if !defined(_PATH_PROCNET_SCTP_ASSOCS)
89e4517
+#define	_PATH_PROCNET_SCTP_ASSOCS	"/proc/net/sctp/assocs"
89e4517
+#endif
89e4517
+    INFO_GUTS(_PATH_PROCNET_SCTP_ASSOCS, "AF INET (sctp)",
89e4517
+              sctp_assoc_do_one);
89e4517
+}
89e4517
+
89e4517
+static int sctp_info(void)
89e4517
+{
89e4517
+    if(flag_all)
89e4517
+    	sctp_info_eps();
89e4517
+    return sctp_info_assocs();
89e4517
+}
89e4517
+
89e4517
 static void raw_do_one(int lnr, const char *line)
89e4517
 {
89e4517
     char buffer[8192], local_addr[64], rem_addr[64];
89e4517
@@ -1742,7 +2126,7 @@
89e4517
     fprintf(stderr, _("        -Z, --context              display SELinux security context for sockets\n\n"));
89e4517
 
89e4517
     fprintf(stderr, _("  <Iface>: Name of interface to monitor/list.\n"));
89e4517
-    fprintf(stderr, _("  <Socket>={-t|--tcp} {-u|--udp} {-w|--raw} {-x|--unix} --ax25 --ipx --netrom\n"));
89e4517
+    fprintf(stderr, _("  <Socket>={-t|--tcp} {-u|--udp} {-S|--sctp} {-w|--raw} {-x|--unix} --ax25 --ipx --netrom\n"));
89e4517
     fprintf(stderr, _("  <AF>=Use '-A <af>' or '--<af>'; default: %s\n"), DFLT_AF);
89e4517
     fprintf(stderr, _("  List of possible address families (which support routing):\n"));
89e4517
     print_aflist(1); /* 1 = routeable */
89e4517
@@ -1769,6 +2153,7 @@
89e4517
 	{"protocol", 1, 0, 'A'},
89e4517
 	{"tcp", 0, 0, 't'},
89e4517
 	{"udp", 0, 0, 'u'},
89e4517
+	{"sctp", 0, 0, 'S' },
89e4517
 	{"raw", 0, 0, 'w'},
89e4517
 	{"unix", 0, 0, 'x'},
89e4517
 	{"listening", 0, 0, 'l'},
89e4517
@@ -1801,7 +2186,7 @@
89e4517
 
89e4517
     afname[0] = '\0';
89e4517
 
89e4517
-    while ((i = getopt_long(argc, argv, "MCFA:acdegphiI::nNorstuVv?wxlZT", longopts, &lop)) != EOF)
89e4517
+    while ((i = getopt_long(argc, argv, "MCFA:acdegphiI::nNorstuSVv?wxlZT", longopts, &lop)) != EOF)
89e4517
 	switch (i) {
89e4517
 	case -1:
89e4517
 	    break;
89e4517
@@ -1887,10 +2272,12 @@
89e4517
 	case 't':
89e4517
 	    flag_tcp++;
89e4517
 	    break;
89e4517
-
89e4517
 	case 'u':
89e4517
 	    flag_udp++;
89e4517
 	    break;
89e4517
+	case 'S':
89e4517
+	    flag_sctp++;
89e4517
+	    break;
89e4517
 	case 'w':
89e4517
 	    flag_raw++;
89e4517
 	    break;
89e4517
@@ -1932,13 +2319,13 @@
89e4517
     if (flag_int + flag_rou + flag_mas + flag_sta > 1)
89e4517
 	usage();
89e4517
 
89e4517
-    if ((flag_inet || flag_inet6 || flag_sta) && !(flag_tcp || flag_udp || flag_raw))
89e4517
-	flag_tcp = flag_udp = flag_raw = 1;
89e4517
+    if ((flag_inet || flag_inet6 || flag_sta) && !(flag_tcp || flag_udp || flag_sctp || flag_raw))
89e4517
+	flag_tcp = flag_udp = flag_sctp = flag_raw = 1;
89e4517
 
89e4517
-    if ((flag_tcp || flag_udp || flag_raw || flag_igmp) && !(flag_inet || flag_inet6))
89e4517
+    if ((flag_tcp || flag_udp || flag_sctp || flag_raw || flag_igmp) && !(flag_inet || flag_inet6))
89e4517
         flag_inet = flag_inet6 = 1;
89e4517
 
89e4517
-    flag_arg = flag_tcp + flag_udp + flag_raw + flag_unx + flag_ipx
89e4517
+    flag_arg = flag_tcp + flag_udp + flag_sctp + flag_raw + flag_unx + flag_ipx
89e4517
 	+ flag_ax25 + flag_netrom + flag_igmp + flag_x25;
89e4517
     
89e4517
     if (flag_mas) {
89e4517
@@ -1964,7 +2351,7 @@
89e4517
     if (flag_sta) {
89e4517
       for(;;) {
89e4517
         inittab();
89e4517
-	i = parsesnmp(flag_raw, flag_tcp, flag_udp);
89e4517
+	i = parsesnmp(flag_raw, flag_tcp, flag_udp, flag_sctp);
89e4517
 	
89e4517
 	if(i || !flag_cnt)
89e4517
 	  break;
89e4517
@@ -2006,7 +2393,7 @@
89e4517
 	return (i);
89e4517
     }
89e4517
     for (;;) {
89e4517
-	if (!flag_arg || flag_tcp || flag_udp || flag_raw) {
89e4517
+	if (!flag_arg || flag_tcp || flag_udp || flag_sctp || flag_raw) {
89e4517
 #if HAVE_AFINET
89e4517
 	    prg_cache_load();
89e4517
 	    printf(_("Active Internet connections "));	/* xxx */
89e4517
@@ -2044,6 +2431,11 @@
89e4517
 	    if (i)
89e4517
 		return (i);
89e4517
 	}
89e4517
+	if (!flag_arg || flag_sctp) {
89e4517
+	    i = sctp_info();
89e4517
+	    if (i)
89e4517
+		return (i);
89e4517
+	}
89e4517
 	if (!flag_arg || flag_raw) {
89e4517
 	    i = raw_info();
89e4517
 	    if (i)