⭐ 欢迎来到虫虫下载站! | 📦 资源下载 📁 资源专辑 ℹ️ 关于我们
⭐ 虫虫下载站

📄 print-pgm.c

📁 TCPDUMP的C语言源代码,是在数据链路层的应用
💻 C
📖 第 1 页 / 共 2 页
字号:
/* * Redistribution and use in source and binary forms, with or without * modification, are permitted provided that: (1) source code * distributions retain the above copyright notice and this paragraph * in its entirety, and (2) distributions including binary code include * the above copyright notice and this paragraph in its entirety in * the documentation or other materials provided with the distribution. * THIS SOFTWARE IS PROVIDED ``AS IS'' AND * WITHOUT ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, WITHOUT * LIMITATION, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS * FOR A PARTICULAR PURPOSE. * * Original code by Andy Heffernan (ahh@juniper.net) */#ifndef lintstatic const char rcsid[] _U_ =    "@(#) $Header: /tcpdump/master/tcpdump/print-pgm.c,v 1.5 2005-06-07 22:05:58 guy Exp $";#endif#ifdef HAVE_CONFIG_H#include "config.h"#endif#include <tcpdump-stdinc.h>#include <stdio.h>#include <stdlib.h>#include <string.h>#include "interface.h"#include "extract.h"#include "addrtoname.h"#include "ip.h"#ifdef INET6#include "ip6.h"#endif#include "ipproto.h"/* * PGM header (RFC 3208) */struct pgm_header {    u_int16_t	pgm_sport;    u_int16_t	pgm_dport;    u_int8_t	pgm_type;    u_int8_t	pgm_options;    u_int16_t	pgm_sum;    u_int8_t	pgm_gsid[6];    u_int16_t	pgm_length;};struct pgm_spm {    u_int32_t	pgms_seq;    u_int32_t	pgms_trailseq;    u_int32_t	pgms_leadseq;    u_int16_t	pgms_nla_afi;    u_int16_t	pgms_reserved;    /* ... u_int8_t	pgms_nla[0]; */    /* ... options */};struct pgm_nak {    u_int32_t	pgmn_seq;    u_int16_t	pgmn_source_afi;    u_int16_t	pgmn_reserved;    /* ... u_int8_t	pgmn_source[0]; */    /* ... u_int16_t	pgmn_group_afi */    /* ... u_int16_t	pgmn_reserved2; */    /* ... u_int8_t	pgmn_group[0]; */    /* ... options */};struct pgm_poll {    u_int32_t	pgmp_seq;    u_int16_t	pgmp_round;    u_int16_t	pgmp_reserved;    /* ... options */};struct pgm_polr {    u_int32_t	pgmp_seq;    u_int16_t	pgmp_round;    u_int16_t	pgmp_subtype;    u_int16_t	pgmp_nla_afi;    u_int16_t	pgmp_reserved;    /* ... u_int8_t	pgmp_nla[0]; */    /* ... options */};struct pgm_data {    u_int32_t	pgmd_seq;    u_int32_t	pgmd_trailseq;    /* ... options */};typedef enum _pgm_type {    PGM_SPM = 0,		/* source path message */    PGM_POLL = 1,		/* POLL Request */    PGM_POLR = 2,		/* POLL Response */    PGM_ODATA = 4,		/* original data */    PGM_RDATA = 5,		/* repair data */    PGM_NAK = 8,		/* NAK */    PGM_NULLNAK = 9,		/* Null NAK */    PGM_NCF = 10,		/* NAK Confirmation */    PGM_ACK = 11,		/* ACK for congestion control */    PGM_SPMR = 12,		/* SPM request */    PGM_MAX = 255} pgm_type;#define PGM_OPT_BIT_PRESENT	0x01#define PGM_OPT_BIT_NETWORK	0x02#define PGM_OPT_BIT_VAR_PKTLEN	0x40#define PGM_OPT_BIT_PARITY	0x80#define PGM_OPT_LENGTH		0x00#define PGM_OPT_FRAGMENT        0x01#define PGM_OPT_NAK_LIST        0x02#define PGM_OPT_JOIN            0x03#define PGM_OPT_NAK_BO_IVL	0x04#define PGM_OPT_NAK_BO_RNG	0x05#define PGM_OPT_REDIRECT        0x07#define PGM_OPT_PARITY_PRM      0x08#define PGM_OPT_PARITY_GRP      0x09#define PGM_OPT_CURR_TGSIZE     0x0A#define PGM_OPT_NBR_UNREACH	0x0B#define PGM_OPT_PATH_NLA	0x0C#define PGM_OPT_SYN             0x0D#define PGM_OPT_FIN             0x0E#define PGM_OPT_RST             0x0F#define PGM_OPT_CR		0x10#define PGM_OPT_CRQST		0x11     #define PGM_OPT_MASK		0x7f#define PGM_OPT_END		0x80    /* end of options marker */#define PGM_MIN_OPT_LEN		4#ifndef AFI_IP#define AFI_IP		1#define AFI_IP6	        2#endifvoidpgm_print(register const u_char *bp, register u_int length,	  register const u_char *bp2){	register const struct pgm_header *pgm;	register const struct ip *ip;	register char ch;	u_int16_t sport, dport;	int addr_size;	const void *nla;	int nla_af;#ifdef INET6	char nla_buf[INET6_ADDRSTRLEN];	register const struct ip6_hdr *ip6;#else	char nla_buf[INET_ADDRSTRLEN];#endif	u_int8_t opt_type, opt_len, flags1, flags2;	u_int32_t seq, opts_len, len, offset;	pgm = (struct pgm_header *)bp;	ip = (struct ip *)bp2;#ifdef INET6	if (IP_V(ip) == 6)		ip6 = (struct ip6_hdr *)bp2;	else		ip6 = NULL;#else /* INET6 */	if (IP_V(ip) == 6) {		(void)printf("Can't handle IPv6");		return;	}#endif /* INET6 */	ch = '\0';	if (!TTEST(pgm->pgm_dport)) {#ifdef INET6		if (ip6) {			(void)printf("%s > %s: [|pgm]",				ip6addr_string(&ip6->ip6_src),				ip6addr_string(&ip6->ip6_dst));			return;		} else#endif /* INET6 */		{			(void)printf("%s > %s: [|pgm]",				ipaddr_string(&ip->ip_src),				ipaddr_string(&ip->ip_dst));			return;		}	}	sport = EXTRACT_16BITS(&pgm->pgm_sport);	dport = EXTRACT_16BITS(&pgm->pgm_dport);#ifdef INET6	if (ip6) {		if (ip6->ip6_nxt == IPPROTO_PGM) {			(void)printf("%s.%s > %s.%s: ",				ip6addr_string(&ip6->ip6_src),				tcpport_string(sport),				ip6addr_string(&ip6->ip6_dst),				tcpport_string(dport));		} else {			(void)printf("%s > %s: ",				tcpport_string(sport), tcpport_string(dport));		}	} else#endif /*INET6*/	{		if (ip->ip_p == IPPROTO_PGM) {			(void)printf("%s.%s > %s.%s: ",				ipaddr_string(&ip->ip_src),				tcpport_string(sport),				ipaddr_string(&ip->ip_dst),				tcpport_string(dport));		} else {			(void)printf("%s > %s: ",				tcpport_string(sport), tcpport_string(dport));		}	}	TCHECK(*pgm);        (void)printf("PGM, length %u", pgm->pgm_length);        if (!vflag)            return;        if (length > pgm->pgm_length)            length = pgm->pgm_length;	(void)printf(" 0x%02x%02x%02x%02x%02x%02x ",		     pgm->pgm_gsid[0],                     pgm->pgm_gsid[1],                     pgm->pgm_gsid[2],		     pgm->pgm_gsid[3],                     pgm->pgm_gsid[4],                     pgm->pgm_gsid[5]);	switch (pgm->pgm_type) {	case PGM_SPM: {	    struct pgm_spm *spm;	    spm = (struct pgm_spm *)(pgm + 1);	    TCHECK(*spm);	    switch (EXTRACT_16BITS(&spm->pgms_nla_afi)) {	    case AFI_IP:		addr_size = sizeof(struct in_addr);		nla_af = AF_INET;		break;#ifdef INET6	    case AFI_IP6:		addr_size = sizeof(struct in6_addr);		nla_af = AF_INET6;		break;#endif	    default:		goto trunc;		break;	    }	    bp = (u_char *) (spm + 1);	    TCHECK2(*bp, addr_size);	    nla = bp;	    bp += addr_size;	    inet_ntop(nla_af, nla, nla_buf, sizeof(nla_buf));	    (void)printf("SPM seq %u trail %u lead %u nla %s",			 EXTRACT_32BITS(&spm->pgms_seq),                         EXTRACT_32BITS(&spm->pgms_trailseq),			 EXTRACT_32BITS(&spm->pgms_leadseq),                         nla_buf);	    break;	}	case PGM_POLL: {	    struct pgm_poll *poll;	    poll = (struct pgm_poll *)(pgm + 1);	    TCHECK(*poll);	    (void)printf("POLL seq %u round %u",			 EXTRACT_32BITS(&poll->pgmp_seq),                         EXTRACT_16BITS(&poll->pgmp_round));	    bp = (u_char *) (poll + 1);	    break;	}	case PGM_POLR: {	    struct pgm_polr *polr;	    u_int32_t ivl, rnd, mask;	    polr = (struct pgm_polr *)(pgm + 1);	    TCHECK(*polr);	    switch (EXTRACT_16BITS(&polr->pgmp_nla_afi)) {	    case AFI_IP:		addr_size = sizeof(struct in_addr);		nla_af = AF_INET;		break;#ifdef INET6	    case AFI_IP6:		addr_size = sizeof(struct in6_addr);		nla_af = AF_INET6;		break;#endif	    default:		goto trunc;		break;	    }	    bp = (u_char *) (polr + 1);	    TCHECK2(*bp, addr_size);	    nla = bp;	    bp += addr_size;	    inet_ntop(nla_af, nla, nla_buf, sizeof(nla_buf));	    TCHECK2(*bp, sizeof(u_int32_t));	    ivl = EXTRACT_32BITS(bp);	    bp += sizeof(u_int32_t);	    TCHECK2(*bp, sizeof(u_int32_t));	    rnd = EXTRACT_32BITS(bp);	    bp += sizeof(u_int32_t);	    TCHECK2(*bp, sizeof(u_int32_t));	    mask = EXTRACT_32BITS(bp);	    bp += sizeof(u_int32_t);	    (void)printf("POLR seq %u round %u nla %s ivl %u rnd 0x%08x "			 "mask 0x%08x", EXTRACT_32BITS(&polr->pgmp_seq),			 EXTRACT_16BITS(&polr->pgmp_round), nla_buf, ivl, rnd, mask);	    break;	}	case PGM_ODATA: {	    struct pgm_data *odata;	    odata = (struct pgm_data *)(pgm + 1);	    TCHECK(*odata);	    (void)printf("ODATA trail %u seq %u",			 EXTRACT_32BITS(&odata->pgmd_trailseq),			 EXTRACT_32BITS(&odata->pgmd_seq));	    bp = (u_char *) (odata + 1);	    break;	}	case PGM_RDATA: {	    struct pgm_data *rdata;	    rdata = (struct pgm_data *)(pgm + 1);	    TCHECK(*rdata);	    (void)printf("RDATA trail %u seq %u",			 EXTRACT_32BITS(&rdata->pgmd_trailseq),			 EXTRACT_32BITS(&rdata->pgmd_seq));	    bp = (u_char *) (rdata + 1);	    break;	}	case PGM_NAK:	case PGM_NULLNAK:	case PGM_NCF: {	    struct pgm_nak *nak;	    const void *source, *group;	    int source_af, group_af;#ifdef INET6	    char source_buf[INET6_ADDRSTRLEN], group_buf[INET6_ADDRSTRLEN];#else	    char source_buf[INET_ADDRSTRLEN], group_buf[INET_ADDRSTRLEN];#endif	    nak = (struct pgm_nak *)(pgm + 1);	    TCHECK(*nak);	    /*	     * Skip past the source, saving info along the way	     * and stopping if we don't have enough.

⌨️ 快捷键说明

复制代码 Ctrl + C
搜索代码 Ctrl + F
全屏模式 F11
切换主题 Ctrl + Shift + D
显示快捷键 ?
增大字号 Ctrl + =
减小字号 Ctrl + -