1 /* $KAME: pfkey_dump.c,v 1.45 2003/09/08 10:14:56 itojun Exp $ */
4 * Copyright (C) 1995, 1996, 1997, 1998, and 1999 WIDE Project.
7 * Redistribution and use in source and binary forms, with or without
8 * modification, are permitted provided that the following conditions
10 * 1. Redistributions of source code must retain the above copyright
11 * notice, this list of conditions and the following disclaimer.
12 * 2. Redistributions in binary form must reproduce the above copyright
13 * notice, this list of conditions and the following disclaimer in the
14 * documentation and/or other materials provided with the distribution.
15 * 3. Neither the name of the project nor the names of its contributors
16 * may be used to endorse or promote products derived from this software
17 * without specific prior written permission.
19 * THIS SOFTWARE IS PROVIDED BY THE PROJECT AND CONTRIBUTORS ``AS IS'' AND
20 * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
21 * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
22 * ARE DISCLAIMED. IN NO EVENT SHALL THE PROJECT OR CONTRIBUTORS BE LIABLE
23 * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
24 * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
25 * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
26 * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
27 * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
28 * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
36 #include <sys/types.h>
37 #include <sys/param.h>
38 #include <sys/socket.h>
39 #ifdef HAVE_NETINET6_IPSEC
40 # include <netinet6/ipsec.h>
42 # include <netinet/ipsec.h>
45 #include <System/net/pfkeyv2.h>
47 #include <netinet/in.h>
48 #include <arpa/inet.h>
57 #include "ipsec_strerror.h"
60 /* cope with old kame headers - ugly */
61 #ifndef SADB_X_AALG_MD5
62 #define SADB_X_AALG_MD5 SADB_AALG_MD5
64 #ifndef SADB_X_AALG_SHA
65 #define SADB_X_AALG_SHA SADB_AALG_SHA
67 #ifndef SADB_X_AALG_NULL
68 #define SADB_X_AALG_NULL SADB_AALG_NULL
71 #ifndef SADB_X_EALG_BLOWFISHCBC
72 #define SADB_X_EALG_BLOWFISHCBC SADB_EALG_BLOWFISHCBC
74 #ifndef SADB_X_EALG_CAST128CBC
75 #define SADB_X_EALG_CAST128CBC SADB_EALG_CAST128CBC
77 #ifndef SADB_X_EALG_RC5CBC
78 #ifdef SADB_EALG_RC5CBC
79 #define SADB_X_EALG_RC5CBC SADB_EALG_RC5CBC
83 #define GETMSGSTR(str, num) \
86 if (sizeof((str)[0]) == 0 \
87 || num >= sizeof(str)/sizeof((str)[0])) \
88 printf("%u ", (num)); \
89 else if (strlen((str)[(num)]) == 0) \
90 printf("%u ", (num)); \
92 printf("%s ", (str)[(num)]); \
93 } while (/*CONSTCOND*/0)
95 #define GETMSGV2S(v2s, num) \
98 for (p = (v2s); p && p->str; p++) { \
99 if (p->val == (num)) \
103 printf("%s ", p->str); \
105 printf("%u ", (num)); \
106 } while (/*CONSTCOND*/0)
108 static char *str_ipaddr
__P((struct sockaddr
*));
109 static char *str_ipport
__P((struct sockaddr
*));
110 static char *str_prefport
__P((u_int
, u_int
, u_int
, u_int
));
111 static void str_upperspec
__P((u_int
, u_int
, u_int
));
112 static char *str_time
__P((time_t));
113 static void str_lifetime_byte
__P((struct sadb_lifetime
*, char *));
114 static void pfkey_sadump1(struct sadb_msg
*, int);
115 static void pfkey_spdump1(struct sadb_msg
*, int);
123 * Must to be re-written about following strings.
125 static char *str_satype
[] = {
140 static char *str_mode
[] = {
146 static char *str_state
[] = {
153 static struct val2str str_alg_auth
[] = {
154 { SADB_AALG_NONE
, "none", },
155 { SADB_AALG_MD5HMAC
, "hmac-md5", },
156 { SADB_AALG_SHA1HMAC
, "hmac-sha1", },
157 { SADB_X_AALG_MD5
, "md5", },
158 { SADB_X_AALG_SHA
, "sha", },
159 { SADB_X_AALG_NULL
, "null", },
160 #ifdef SADB_X_AALG_TCP_MD5
161 { SADB_X_AALG_TCP_MD5
, "tcp-md5", },
163 #ifdef SADB_X_AALG_SHA2_256
164 { SADB_X_AALG_SHA2_256
, "hmac-sha256", },
166 #ifdef SADB_X_AALG_SHA2_384
167 { SADB_X_AALG_SHA2_384
, "hmac-sha384", },
169 #ifdef SADB_X_AALG_SHA2_512
170 { SADB_X_AALG_SHA2_512
, "hmac-sha512", },
172 #ifdef SADB_X_AALG_RIPEMD160HMAC
173 { SADB_X_AALG_RIPEMD160HMAC
, "hmac-ripemd160", },
175 #ifdef SADB_X_AALG_AES_XCBC_MAC
176 { SADB_X_AALG_AES_XCBC_MAC
, "aes-xcbc-mac", },
181 static struct val2str str_alg_enc
[] = {
182 { SADB_EALG_NONE
, "none", },
183 { SADB_EALG_DESCBC
, "des-cbc", },
184 { SADB_EALG_3DESCBC
, "3des-cbc", },
185 { SADB_EALG_NULL
, "null", },
186 #ifdef SADB_X_EALG_RC5CBC
187 { SADB_X_EALG_RC5CBC
, "rc5-cbc", },
189 { SADB_X_EALG_CAST128CBC
, "cast128-cbc", },
190 { SADB_X_EALG_BLOWFISHCBC
, "blowfish-cbc", },
191 #ifdef SADB_X_EALG_AESCBC
192 { SADB_X_EALG_AESCBC
, "aes-cbc", },
194 #ifdef SADB_X_EALG_TWOFISHCBC
195 { SADB_X_EALG_TWOFISHCBC
, "twofish-cbc", },
197 #ifdef SADB_X_EALG_AESCTR
198 { SADB_X_EALG_AESCTR
, "aes-ctr", },
203 static struct val2str str_alg_comp
[] = {
204 { SADB_X_CALG_NONE
, "none", },
205 { SADB_X_CALG_OUI
, "oui", },
206 { SADB_X_CALG_DEFLATE
, "deflate", },
207 { SADB_X_CALG_LZS
, "lzs", },
212 * dump SADB_MSG formated. For debugging, you should use kdebug_sadb().
223 pfkey_sadump_withports(m
)
230 pfkey_sadump1(m
, withports
)
234 caddr_t mhp
[SADB_EXT_MAX
+ 1];
235 struct sadb_sa
*m_sa
;
236 struct sadb_x_sa2
*m_sa2
;
237 struct sadb_lifetime
*m_lftc
, *m_lfth
, *m_lfts
;
238 struct sadb_address
*m_saddr
, *m_daddr
;
240 struct sadb_address
*m_paddr
;
242 struct sadb_key
*m_auth
, *m_enc
;
244 struct sadb_ident
*m_sid
, *m_did
;
245 struct sadb_sens
*m_sens
;
247 #ifdef SADB_X_EXT_NAT_T_TYPE
248 struct sadb_x_nat_t_type
*natt_type
;
249 struct sadb_x_nat_t_port
*natt_sport
, *natt_dport
;
250 struct sadb_address
*natt_oa
;
256 /* check pfkey message. */
257 if (pfkey_align(m
, mhp
)) {
258 printf("%s\n", ipsec_strerror());
261 if (pfkey_check(mhp
)) {
262 printf("%s\n", ipsec_strerror());
266 m_sa
= (void *)mhp
[SADB_EXT_SA
];
267 m_sa2
= (void *)mhp
[SADB_X_EXT_SA2
];
268 m_lftc
= (void *)mhp
[SADB_EXT_LIFETIME_CURRENT
];
269 m_lfth
= (void *)mhp
[SADB_EXT_LIFETIME_HARD
];
270 m_lfts
= (void *)mhp
[SADB_EXT_LIFETIME_SOFT
];
271 m_saddr
= (void *)mhp
[SADB_EXT_ADDRESS_SRC
];
272 m_daddr
= (void *)mhp
[SADB_EXT_ADDRESS_DST
];
274 m_paddr
= (void *)mhp
[SADB_EXT_ADDRESS_PROXY
];
276 m_auth
= (void *)mhp
[SADB_EXT_KEY_AUTH
];
277 m_enc
= (void *)mhp
[SADB_EXT_KEY_ENCRYPT
];
279 m_sid
= (void *)mhp
[SADB_EXT_IDENTITY_SRC
];
280 m_did
= (void *)mhp
[SADB_EXT_IDENTITY_DST
];
281 m_sens
= (void *)mhp
[SADB_EXT_SENSITIVITY
];
283 #ifdef SADB_X_EXT_NAT_T_TYPE
284 natt_type
= (void *)mhp
[SADB_X_EXT_NAT_T_TYPE
];
285 natt_sport
= (void *)mhp
[SADB_X_EXT_NAT_T_SPORT
];
286 natt_dport
= (void *)mhp
[SADB_X_EXT_NAT_T_DPORT
];
287 natt_oa
= (void *)mhp
[SADB_X_EXT_NAT_T_OA
];
289 if (natt_type
&& natt_type
->sadb_x_nat_t_type_type
)
293 if (m_saddr
== NULL
) {
294 printf("no ADDRESS_SRC extension.\n");
297 sa
= (void *)(m_saddr
+ 1);
299 printf("%s[%s]", str_ipaddr(sa
), str_ipport(sa
));
301 printf("%s", str_ipaddr(sa
));
302 #ifdef SADB_X_EXT_NAT_T_TYPE
303 if (use_natt
&& natt_sport
)
304 printf("[%u]", ntohs(natt_sport
->sadb_x_nat_t_port_port
));
308 /* destination address */
309 if (m_daddr
== NULL
) {
310 printf(" no ADDRESS_DST extension.\n");
313 sa
= (void *)(m_daddr
+ 1);
315 printf("%s[%s]", str_ipaddr(sa
), str_ipport(sa
));
317 printf("%s", str_ipaddr(sa
));
318 #ifdef SADB_X_EXT_NAT_T_TYPE
319 if (use_natt
&& natt_dport
)
320 printf("[%u]", ntohs(natt_dport
->sadb_x_nat_t_port_port
));
326 printf("no SA extension.\n");
330 printf("no SA2 extension.\n");
335 #ifdef SADB_X_EXT_NAT_T_TYPE
336 if (use_natt
&& m
->sadb_msg_satype
== SADB_SATYPE_ESP
)
341 if (!use_natt
|| m
->sadb_msg_satype
!= SADB_SATYPE_ESP
)
343 GETMSGSTR(str_satype
, m
->sadb_msg_satype
);
346 GETMSGSTR(str_mode
, m_sa2
->sadb_x_sa2_mode
);
348 printf("spi=%u(0x%08x) reqid=%u(0x%08x)\n",
349 (u_int32_t
)ntohl(m_sa
->sadb_sa_spi
),
350 (u_int32_t
)ntohl(m_sa
->sadb_sa_spi
),
351 (u_int32_t
)m_sa2
->sadb_x_sa2_reqid
,
352 (u_int32_t
)m_sa2
->sadb_x_sa2_reqid
);
354 #ifdef SADB_X_EXT_NAT_T_TYPE
355 /* other NAT-T information */
356 if (use_natt
&& natt_oa
)
357 printf("\tNAT OA=%s\n",
358 str_ipaddr((void *)(natt_oa
+ 1)));
362 if (m
->sadb_msg_satype
== SADB_X_SATYPE_IPCOMP
) {
364 GETMSGV2S(str_alg_comp
, m_sa
->sadb_sa_encrypt
);
365 } else if (m
->sadb_msg_satype
== SADB_SATYPE_ESP
) {
368 GETMSGV2S(str_alg_enc
, m_sa
->sadb_sa_encrypt
);
369 ipsec_hexdump((caddr_t
)(void *)m_enc
+ sizeof(*m_enc
),
370 m_enc
->sadb_key_bits
/ 8);
375 /* authentication key */
376 if (m_auth
!= NULL
) {
378 GETMSGV2S(str_alg_auth
, m_sa
->sadb_sa_auth
);
379 ipsec_hexdump((caddr_t
)(void *)m_auth
+ sizeof(*m_auth
),
380 m_auth
->sadb_key_bits
/ 8);
384 /* replay windoe size & flags */
385 printf("\tseq=0x%08x replay=%u flags=0x%08x ",
386 m_sa2
->sadb_x_sa2_sequence
,
387 m_sa
->sadb_sa_replay
,
388 m_sa
->sadb_sa_flags
);
392 GETMSGSTR(str_state
, m_sa
->sadb_sa_state
);
396 if (m_lftc
!= NULL
) {
397 time_t tmp_time
= time(0);
399 printf("\tcreated: %s",
400 str_time((long)m_lftc
->sadb_lifetime_addtime
));
401 printf("\tcurrent: %s\n", str_time(tmp_time
));
402 printf("\tdiff: %lu(s)",
403 (u_long
)(m_lftc
->sadb_lifetime_addtime
== 0 ?
404 0 : (tmp_time
- m_lftc
->sadb_lifetime_addtime
)));
406 printf("\thard: %lu(s)",
407 (u_long
)(m_lfth
== NULL
?
408 0 : m_lfth
->sadb_lifetime_addtime
));
409 printf("\tsoft: %lu(s)\n",
410 (u_long
)(m_lfts
== NULL
?
411 0 : m_lfts
->sadb_lifetime_addtime
));
414 str_time((long)m_lftc
->sadb_lifetime_usetime
));
415 printf("\thard: %lu(s)",
416 (u_long
)(m_lfth
== NULL
?
417 0 : m_lfth
->sadb_lifetime_usetime
));
418 printf("\tsoft: %lu(s)\n",
419 (u_long
)(m_lfts
== NULL
?
420 0 : m_lfts
->sadb_lifetime_usetime
));
422 str_lifetime_byte(m_lftc
, "current");
423 str_lifetime_byte(m_lfth
, "hard");
424 str_lifetime_byte(m_lfts
, "soft");
427 printf("\tallocated: %lu",
428 (unsigned long)m_lftc
->sadb_lifetime_allocations
);
429 printf("\thard: %lu",
430 (u_long
)(m_lfth
== NULL
?
431 0 : m_lfth
->sadb_lifetime_allocations
));
432 printf("\tsoft: %lu\n",
433 (u_long
)(m_lfts
== NULL
?
434 0 : m_lfts
->sadb_lifetime_allocations
));
437 printf("\tsadb_seq=%lu pid=%lu ",
438 (u_long
)m
->sadb_msg_seq
,
439 (u_long
)m
->sadb_msg_pid
);
442 printf("refcnt=%u\n", m
->sadb_msg_reserved
);
455 pfkey_spdump_withports(m
)
462 pfkey_spdump1(m
, withports
)
466 char pbuf
[NI_MAXSERV
];
467 caddr_t mhp
[SADB_EXT_MAX
+ 1];
468 struct sadb_address
*m_saddr
, *m_daddr
;
469 #ifdef SADB_X_EXT_TAG
470 struct sadb_x_tag
*m_tag
;
472 struct sadb_x_policy
*m_xpl
;
473 struct sadb_lifetime
*m_lftc
= NULL
, *m_lfth
= NULL
;
475 u_int16_t sport
= 0, dport
= 0;
477 /* check pfkey message. */
478 if (pfkey_align(m
, mhp
)) {
479 printf("%s\n", ipsec_strerror());
482 if (pfkey_check(mhp
)) {
483 printf("%s\n", ipsec_strerror());
487 m_saddr
= (void *)mhp
[SADB_EXT_ADDRESS_SRC
];
488 m_daddr
= (void *)mhp
[SADB_EXT_ADDRESS_DST
];
489 #ifdef SADB_X_EXT_TAG
490 m_tag
= (void *)mhp
[SADB_X_EXT_TAG
];
492 m_xpl
= (void *)mhp
[SADB_X_EXT_POLICY
];
493 m_lftc
= (void *)mhp
[SADB_EXT_LIFETIME_CURRENT
];
494 m_lfth
= (void *)mhp
[SADB_EXT_LIFETIME_HARD
];
496 if (m_saddr
&& m_daddr
) {
498 sa
= (void *)(m_saddr
+ 1);
499 switch (sa
->sa_family
) {
502 if (getnameinfo(sa
, (socklen_t
)sysdep_sa_len((struct sockaddr
*)sa
), NULL
,
503 0, pbuf
, sizeof(pbuf
), NI_NUMERICSERV
) != 0)
507 printf("%s%s ", str_ipaddr(sa
),
508 str_prefport((u_int
)sa
->sa_family
,
509 (u_int
)m_saddr
->sadb_address_prefixlen
,
511 (u_int
)m_saddr
->sadb_address_proto
));
514 printf("unknown-af ");
518 /* destination address */
519 sa
= (void *)(m_daddr
+ 1);
520 switch (sa
->sa_family
) {
523 if (getnameinfo(sa
, (socklen_t
)sysdep_sa_len((struct sockaddr
*)sa
), NULL
,
524 0, pbuf
, sizeof(pbuf
), NI_NUMERICSERV
) != 0)
528 printf("%s%s ", str_ipaddr(sa
),
529 str_prefport((u_int
)sa
->sa_family
,
530 (u_int
)m_daddr
->sadb_address_prefixlen
,
532 (u_int
)m_saddr
->sadb_address_proto
));
535 printf("unknown-af ");
539 /* upper layer protocol */
540 if (m_saddr
->sadb_address_proto
!=
541 m_daddr
->sadb_address_proto
) {
542 printf("upper layer protocol mismatched.\n");
545 str_upperspec((u_int
)m_saddr
->sadb_address_proto
, (u_int
)sport
,
548 #ifdef SADB_X_EXT_TAG
550 printf("tagged \"%s\" ", m_tag
->sadb_x_tag_name
);
553 printf("(no selector, probably per-socket policy) ");
560 printf("no X_POLICY extension.\n");
564 d_xpl
= ipsec_dump_policy_withports(m_xpl
, "\n\t");
566 d_xpl
= ipsec_dump_policy((ipsec_policy_t
)m_xpl
, "\n\t");
569 printf("\n\tPolicy:[%s]\n", ipsec_strerror());
572 printf("\n\t%s\n", d_xpl
);
579 printf("\tcreated: %s ",
580 str_time((long)m_lftc
->sadb_lifetime_addtime
));
581 printf("lastused: %s\n",
582 str_time((long)m_lftc
->sadb_lifetime_usetime
));
585 printf("\tlifetime: %lu(s) ",
586 (u_long
)m_lfth
->sadb_lifetime_addtime
);
587 printf("validtime: %lu(s)\n",
588 (u_long
)m_lfth
->sadb_lifetime_usetime
);
592 printf("\tspid=%ld seq=%ld pid=%ld\n",
593 (u_long
)m_xpl
->sadb_x_policy_id
,
594 (u_long
)m
->sadb_msg_seq
,
595 (u_long
)m
->sadb_msg_pid
);
598 printf("\trefcnt=%u\n", m
->sadb_msg_reserved
);
604 * set "ipaddress" to buffer.
610 static char buf
[NI_MAXHOST
];
611 const int niflag
= NI_NUMERICHOST
;
616 if (getnameinfo(sa
, (socklen_t
)sysdep_sa_len(sa
), buf
, sizeof(buf
),
617 NULL
, 0, niflag
) == 0)
623 * set "port" to buffer.
629 static char buf
[NI_MAXHOST
];
630 const int niflag
= NI_NUMERICSERV
;
635 if (getnameinfo(sa
, (socklen_t
)sysdep_sa_len(sa
), NULL
, 0,
636 buf
, sizeof(buf
), niflag
) == 0)
643 * set "/prefix[port number]" to buffer.
646 str_prefport(family
, pref
, port
, ulp
)
647 u_int family
, pref
, port
, ulp
;
649 static char buf
[128];
656 plen
= sizeof(struct in_addr
) << 3;
659 plen
= sizeof(struct in6_addr
) << 3;
668 snprintf(prefbuf
, sizeof(prefbuf
), "/%u", pref
);
670 if (ulp
== IPPROTO_ICMPV6
)
671 memset(portbuf
, 0, sizeof(portbuf
));
673 if (port
== IPSEC_PORT_ANY
)
674 snprintf(portbuf
, sizeof(portbuf
), "[%s]", "any");
676 snprintf(portbuf
, sizeof(portbuf
), "[%u]", port
);
679 snprintf(buf
, sizeof(buf
), "%s%s", prefbuf
, portbuf
);
685 str_upperspec(ulp
, p1
, p2
)
688 if (ulp
== IPSEC_ULPROTO_ANY
)
690 else if (ulp
== IPPROTO_ICMPV6
) {
692 if (!(p1
== IPSEC_PORT_ANY
&& p2
== IPSEC_PORT_ANY
))
693 printf(" %u,%u", p1
, p2
);
695 struct protoent
*ent
;
702 ent
= getprotobynumber((int)ulp
);
704 printf("%s", ent
->p_name
);
715 * set "Mon Day Time Year" to buffer
721 static char buf
[128];
725 for (;i
< 20;) buf
[i
++] = ' ';
729 memcpy(buf
, t0
+ 4, 20);
738 str_lifetime_byte(x
, str
)
739 struct sadb_lifetime
*x
;
747 printf("\t%s: 0(bytes)", str
);
752 if ((x
->sadb_lifetime_bytes
) / 1024 / 1024) {
753 y
= (x
->sadb_lifetime_bytes
) * 1.0 / 1024 / 1024;
756 } else if ((x
->sadb_lifetime_bytes
) / 1024) {
757 y
= (x
->sadb_lifetime_bytes
) * 1.0 / 1024;
761 y
= (x
->sadb_lifetime_bytes
) * 1.0;
766 y
= (x
->sadb_lifetime_bytes
) * 1.0;
770 printf("\t%s: %.*f(%sbytes)", str
, w
, y
, unit
);