|
version 1.1.1.1, 2012/02/21 22:14:23
|
version 1.1.1.3, 2023/09/27 11:11:38
|
|
Line 1
|
Line 1
|
| /* |
/* |
| * $Id$ |
|
| * |
|
| * libnet |
* libnet |
| * libnet_write.c - writes a prebuilt packet to the network |
* libnet_write.c - writes a prebuilt packet to the network |
| * |
* |
| * Copyright (c) 1998 - 2004 Mike D. Schiffman <mike@infonexus.com> |
* Copyright (c) 1998 - 2004 Mike D. Schiffman <mike@infonexus.com> |
| * All rights reserved. |
* All rights reserved. |
| * win32 specific code | * win32 specific code |
| * Copyright (c) 2002 - 2003 Roberto Larcher <roberto.larcher@libero.it> |
* Copyright (c) 2002 - 2003 Roberto Larcher <roberto.larcher@libero.it> |
| * |
* |
| * Redistribution and use in source and binary forms, with or without |
* Redistribution and use in source and binary forms, with or without |
|
Line 32
|
Line 30
|
| * |
* |
| */ |
*/ |
| |
|
| #if (HAVE_CONFIG_H) | #include "common.h" |
| #include "../include/config.h" | |
| #endif | |
| #if (!(_WIN32) || (__CYGWIN__)) | |
| #include "../include/libnet.h" | |
| #else | |
| #include "../include/win32/libnet.h" | |
| #include "../include/win32/config.h" | |
| #include "packet32.h" | |
| #include "Ntddndis.h" | |
| #endif | |
| |
|
| int |
int |
| libnet_write(libnet_t *l) |
libnet_write(libnet_t *l) |
| { |
{ |
| int c; |
int c; |
| u_int32_t len; | uint32_t len; |
| u_int8_t *packet = NULL; | uint8_t *packet = NULL; |
| |
|
| if (l == NULL) |
if (l == NULL) |
| { | { |
| return (-1); |
return (-1); |
| } |
} |
| |
|
|
Line 72 libnet_write(libnet_t *l)
|
Line 60 libnet_write(libnet_t *l)
|
| if (len > LIBNET_MAX_PACKET) |
if (len > LIBNET_MAX_PACKET) |
| { |
{ |
| snprintf(l->err_buf, LIBNET_ERRBUF_SIZE, |
snprintf(l->err_buf, LIBNET_ERRBUF_SIZE, |
| "%s(): packet is too large (%d bytes)\n", | "%s(): packet is too large (%d bytes)", |
| __func__, len); |
__func__, len); |
| goto done; |
goto done; |
| } |
} |
|
Line 88 libnet_write(libnet_t *l)
|
Line 76 libnet_write(libnet_t *l)
|
| break; |
break; |
| default: |
default: |
| snprintf(l->err_buf, LIBNET_ERRBUF_SIZE, |
snprintf(l->err_buf, LIBNET_ERRBUF_SIZE, |
| "%s(): unsuported injection type\n", __func__); | "%s(): unsupported injection type", __func__); |
| goto done; |
goto done; |
| } |
} |
| |
|
|
Line 125 done:
|
Line 113 done:
|
| |
|
| #if defined (__WIN32__) |
#if defined (__WIN32__) |
| libnet_ptag_t |
libnet_ptag_t |
| libnet_win32_build_fake_ethernet(u_int8_t *dst, u_int8_t *src, u_int16_t type, | libnet_win32_build_fake_ethernet (uint8_t *dst, uint8_t *src, uint16_t type, |
| u_int8_t *payload, u_int32_t payload_s, u_int8_t *packet, libnet_t *l, | const uint8_t *payload, uint32_t payload_s, |
| libnet_ptag_t ptag) | uint8_t *packet, libnet_t *l, libnet_ptag_t ptag) |
| { |
{ |
| struct libnet_ethernet_hdr eth_hdr; |
struct libnet_ethernet_hdr eth_hdr; |
| |
|
|
Line 136 libnet_ptag_t ptag)
|
Line 124 libnet_ptag_t ptag)
|
| return (-1); |
return (-1); |
| } |
} |
| |
|
| memset(ð_hdr, 0, sizeof(eth_hdr)); | memset(ð_hdr, 0, sizeof(eth_hdr)); |
| eth_hdr.ether_type = htons(type); |
eth_hdr.ether_type = htons(type); |
| memcpy(eth_hdr.ether_dhost, dst, ETHER_ADDR_LEN); /* destination address */ |
memcpy(eth_hdr.ether_dhost, dst, ETHER_ADDR_LEN); /* destination address */ |
| memcpy(eth_hdr.ether_shost, src, ETHER_ADDR_LEN); /* source address */ |
memcpy(eth_hdr.ether_shost, src, ETHER_ADDR_LEN); /* source address */ |
|
Line 154 libnet_ptag_t ptag)
|
Line 142 libnet_ptag_t ptag)
|
| } |
} |
| |
|
| libnet_ptag_t |
libnet_ptag_t |
| libnet_win32_build_fake_token(u_int8_t *dst, u_int8_t *src, u_int16_t type, | libnet_win32_build_fake_token (uint8_t *dst, uint8_t *src, uint16_t type, |
| u_int8_t *payload, u_int32_t payload_s, u_int8_t *packet, libnet_t *l, | const uint8_t *payload, uint32_t payload_s, |
| libnet_ptag_t ptag) | uint8_t *packet, libnet_t *l, libnet_ptag_t ptag) |
| { |
{ |
| struct libnet_token_ring_hdr token_ring_hdr; |
struct libnet_token_ring_hdr token_ring_hdr; |
| | |
| if (!packet) |
if (!packet) |
| { |
{ |
| return (-1); |
return (-1); |
| } |
} |
| |
|
| memset(&token_ring_hdr, 0, sizeof(token_ring_hdr)); | memset(&token_ring_hdr, 0, sizeof(token_ring_hdr)); |
| token_ring_hdr.token_ring_access_control = 0x10; |
token_ring_hdr.token_ring_access_control = 0x10; |
| token_ring_hdr.token_ring_frame_control = 0x40; |
token_ring_hdr.token_ring_frame_control = 0x40; |
| token_ring_hdr.token_ring_llc_dsap = 0xaa; |
token_ring_hdr.token_ring_llc_dsap = 0xaa; |
| token_ring_hdr.token_ring_llc_ssap = 0xaa; |
token_ring_hdr.token_ring_llc_ssap = 0xaa; |
| token_ring_hdr.token_ring_llc_control_field = 0x03; |
token_ring_hdr.token_ring_llc_control_field = 0x03; |
| token_ring_hdr.token_ring_type = htons(type); |
token_ring_hdr.token_ring_type = htons(type); |
| memcpy(token_ring_hdr.token_ring_dhost, dst, ETHER_ADDR_LEN); | memcpy(token_ring_hdr.token_ring_dhost, dst, ETHER_ADDR_LEN); |
| memcpy(token_ring_hdr.token_ring_shost, libnet_get_hwaddr(l), | memcpy(token_ring_hdr.token_ring_shost, libnet_get_hwaddr(l), |
| ETHER_ADDR_LEN); | ETHER_ADDR_LEN); |
| | |
| if (payload && payload_s) |
if (payload && payload_s) |
| { |
{ |
| /* |
/* |
|
Line 188 libnet_ptag_t ptag)
|
Line 176 libnet_ptag_t ptag)
|
| return (1); |
return (1); |
| } |
} |
| |
|
| |
|
| int |
int |
| libnet_win32_write_raw_ipv4(libnet_t *l, u_int8_t *payload, u_int32_t payload_s) | libnet_win32_write_raw_ipv4(libnet_t *l, const uint8_t *payload, uint32_t payload_s) |
| { | { |
| static BYTE dst[ETHER_ADDR_LEN]; |
static BYTE dst[ETHER_ADDR_LEN]; |
| static BYTE src[ETHER_ADDR_LEN]; |
static BYTE src[ETHER_ADDR_LEN]; |
| |
|
| u_int8_t *packet = NULL; |
|
| u_int32_t packet_s; |
|
| |
|
| LPPACKET lpPacket = NULL; | uint8_t *packet; |
| DWORD remoteip = 0; | uint32_t packet_s; |
| DWORD BytesTransfered; | |
| | DWORD remoteip = 0; |
| NetType type; |
NetType type; |
| struct libnet_ipv4_hdr *ip_hdr = NULL; |
struct libnet_ipv4_hdr *ip_hdr = NULL; |
| | |
| memset(dst, 0, sizeof(dst)); |
memset(dst, 0, sizeof(dst)); |
| memset(src, 0, sizeof(src)); |
memset(src, 0, sizeof(src)); |
| |
|
| packet_s = payload_s + l->link_offset; |
packet_s = payload_s + l->link_offset; |
| packet = (u_int8_t *)malloc(packet_s); | packet = (uint8_t*) alloca(packet_s); |
| if (packet == NULL) | |
| | /* we have to do the IP checksum |
| | * FIXME: warning is correct, checksum modifies its input. |
| | * Fix is to build checksum inside the allocated 'packet' |
| | */ |
| | if (libnet_inet_checksum(l, (uint8_t*)payload, IPPROTO_IP, LIBNET_IPV4_H, payload, payload+payload_s) == -1) |
| { |
{ |
| snprintf(l->err_buf, LIBNET_ERRBUF_SIZE, |
|
| "%s(): failed to allocate packet\n", __func__); |
|
| return (-1); |
|
| } |
|
| |
|
| /* we have to do the IP checksum */ |
|
| if (libnet_do_checksum(l, payload, IPPROTO_IP, LIBNET_IPV4_H) == -1) |
|
| { |
|
| /* error msg set in libnet_do_checksum */ |
/* error msg set in libnet_do_checksum */ |
| return (-1); |
return (-1); |
| } | } |
| |
|
| /* MACs, IPs and other stuff... */ |
/* MACs, IPs and other stuff... */ |
| ip_hdr = (struct libnet_ipv4_hdr *)payload; |
ip_hdr = (struct libnet_ipv4_hdr *)payload; |
| memcpy(src, libnet_get_hwaddr(l), sizeof(src)); |
memcpy(src, libnet_get_hwaddr(l), sizeof(src)); |
| remoteip = ip_hdr->ip_dst.S_un.S_addr; |
remoteip = ip_hdr->ip_dst.S_un.S_addr; |
| | |
| /* check if the remote station is the local station */ |
/* check if the remote station is the local station */ |
| if (remoteip == libnet_get_ipaddr4(l)) |
if (remoteip == libnet_get_ipaddr4(l)) |
| { |
{ |
|
Line 239 libnet_win32_write_raw_ipv4(libnet_t *l, u_int8_t *pay
|
Line 221 libnet_win32_write_raw_ipv4(libnet_t *l, u_int8_t *pay
|
| } |
} |
| |
|
| PacketGetNetType(l->lpAdapter, &type); |
PacketGetNetType(l->lpAdapter, &type); |
| | |
| switch(type.LinkType) |
switch(type.LinkType) |
| { |
{ |
| case NdisMedium802_3: |
case NdisMedium802_3: |
|
Line 257 libnet_win32_write_raw_ipv4(libnet_t *l, u_int8_t *pay
|
Line 239 libnet_win32_write_raw_ipv4(libnet_t *l, u_int8_t *pay
|
| case NdisMediumArcnet878_2: |
case NdisMediumArcnet878_2: |
| default: |
default: |
| snprintf(l->err_buf, LIBNET_ERRBUF_SIZE, |
snprintf(l->err_buf, LIBNET_ERRBUF_SIZE, |
| "%s(): network type (%d) is not supported\n", __func__, | "%s(): network type (%d) is not supported", __func__, |
| type.LinkType); |
type.LinkType); |
| return (-1); |
return (-1); |
| break; |
|
| } |
|
| |
|
| BytesTransfered = -1; |
|
| if ((lpPacket = PacketAllocatePacket()) == NULL) |
|
| { |
|
| snprintf(l->err_buf, LIBNET_ERRBUF_SIZE, |
|
| "%s(): failed to allocate the LPPACKET structure\n", __func__); |
|
| return (-1); |
|
| } |
} |
| | return libnet_write_link (l, packet, packet_s); |
| PacketInitPacket(lpPacket, packet, packet_s); | |
| |
| /* PacketSendPacket returns a BOOLEAN */ | |
| if (PacketSendPacket(l->lpAdapter, lpPacket, TRUE)) | |
| { | |
| BytesTransfered = packet_s; | |
| } | |
| |
| PacketFreePacket(lpPacket); | |
| free(packet); | |
| |
| return (BytesTransfered); | |
| } |
} |
| |
|
| int |
int |
| libnet_write_raw_ipv4(libnet_t *l, u_int8_t *packet, u_int32_t size) | libnet_write_raw_ipv4(libnet_t *l, const uint8_t *packet, uint32_t size) |
| { |
{ |
| return (libnet_win32_write_raw_ipv4(l, packet, size)); |
return (libnet_win32_write_raw_ipv4(l, packet, size)); |
| } |
} |
| |
|
| int |
int |
| libnet_write_raw_ipv6(libnet_t *l, u_int8_t *packet, u_int32_t size) | libnet_write_raw_ipv6(libnet_t *l, const uint8_t *packet, uint32_t size) |
| { |
{ |
| /* no difference in win32 */ |
/* no difference in win32 */ |
| return (libnet_write_raw_ipv4(l, packet, size)); |
return (libnet_write_raw_ipv4(l, packet, size)); |
|
Line 301 libnet_write_raw_ipv6(libnet_t *l, u_int8_t *packet, u
|
Line 262 libnet_write_raw_ipv6(libnet_t *l, u_int8_t *packet, u
|
| #else /* __WIN32__ */ |
#else /* __WIN32__ */ |
| |
|
| int |
int |
| libnet_write_raw_ipv4(libnet_t *l, u_int8_t *packet, u_int32_t size) | libnet_write_raw_ipv4(libnet_t *l, const uint8_t *packet, uint32_t size) |
| { |
{ |
| int c; |
int c; |
| struct sockaddr_in sin; |
struct sockaddr_in sin; |
| struct libnet_ipv4_hdr *ip_hdr; |
struct libnet_ipv4_hdr *ip_hdr; |
| |
|
| if (l == NULL) |
if (l == NULL) |
| { | { |
| return (-1); |
return (-1); |
| } | } |
| |
|
| ip_hdr = (struct libnet_ipv4_hdr *)packet; |
ip_hdr = (struct libnet_ipv4_hdr *)packet; |
| |
|
|
Line 318 libnet_write_raw_ipv4(libnet_t *l, u_int8_t *packet, u
|
Line 279 libnet_write_raw_ipv4(libnet_t *l, u_int8_t *packet, u
|
| /* |
/* |
| * For link access, we don't need to worry about the inconsistencies of |
* For link access, we don't need to worry about the inconsistencies of |
| * certain BSD kernels. However, raw socket nuances abound. Certain |
* certain BSD kernels. However, raw socket nuances abound. Certain |
| * BSD implmentations require the ip_len and ip_off fields to be in host | * BSD implementations require the ip_len and ip_off fields to be in host |
| * byte order. |
* byte order. |
| */ |
*/ |
| ip_hdr->ip_len = FIX(ip_hdr->ip_len); |
ip_hdr->ip_len = FIX(ip_hdr->ip_len); |
|
Line 328 libnet_write_raw_ipv4(libnet_t *l, u_int8_t *packet, u
|
Line 289 libnet_write_raw_ipv4(libnet_t *l, u_int8_t *packet, u
|
| memset(&sin, 0, sizeof(sin)); |
memset(&sin, 0, sizeof(sin)); |
| sin.sin_family = AF_INET; |
sin.sin_family = AF_INET; |
| sin.sin_addr.s_addr = ip_hdr->ip_dst.s_addr; |
sin.sin_addr.s_addr = ip_hdr->ip_dst.s_addr; |
| #if (__WIN32__) |
|
| /* set port for TCP */ |
|
| /* |
|
| * XXX - should first check to see if there's a pblock for a TCP |
|
| * header, if not we can use a dummy value for the port. |
|
| */ |
|
| if (ip_hdr->ip_p == 6) |
|
| { |
|
| struct libnet_tcp_hdr *tcph_p = |
|
| (struct libnet_tcp_hdr *)(packet + (ip_hdr->ip_hl << 2)); |
|
| sin.sin_port = tcph_p->th_dport; |
|
| } |
|
| /* set port for UDP */ |
|
| /* |
|
| * XXX - should first check to see if there's a pblock for a UDP |
|
| * header, if not we can use a dummy value for the port. |
|
| */ |
|
| else if (ip_hdr->ip_p == 17) |
|
| { |
|
| struct libnet_udp_hdr *udph_p = |
|
| (struct libnet_udp_hdr *)(packet + (ip_hdr->ip_hl << 2)); |
|
| sin.sin_port = udph_p->uh_dport; |
|
| } |
|
| #endif /* __WIN32__ */ |
|
| |
|
| c = sendto(l->fd, packet, size, 0, (struct sockaddr *)&sin, |
c = sendto(l->fd, packet, size, 0, (struct sockaddr *)&sin, |
| sizeof(struct sockaddr)); | sizeof(sin)); |
| |
|
| #if (LIBNET_BSD_BYTE_SWAP) |
#if (LIBNET_BSD_BYTE_SWAP) |
| ip_hdr->ip_len = UNFIX(ip_hdr->ip_len); |
ip_hdr->ip_len = UNFIX(ip_hdr->ip_len); |
|
Line 363 libnet_write_raw_ipv4(libnet_t *l, u_int8_t *packet, u
|
Line 300 libnet_write_raw_ipv4(libnet_t *l, u_int8_t *packet, u
|
| |
|
| if (c != size) |
if (c != size) |
| { |
{ |
| #if !(__WIN32__) |
|
| snprintf(l->err_buf, LIBNET_ERRBUF_SIZE, |
snprintf(l->err_buf, LIBNET_ERRBUF_SIZE, |
| "%s(): %d bytes written (%s)\n", __func__, c, | "%s(): %d bytes written (%s)", __func__, c, |
| strerror(errno)); |
strerror(errno)); |
| #else /* __WIN32__ */ |
|
| snprintf(l->err_buf, LIBNET_ERRBUF_SIZE, |
|
| "%s(): %d bytes written (%d)\n", __func__, c, |
|
| WSAGetLastError()); |
|
| #endif /* !__WIN32__ */ |
|
| } |
} |
| return (c); |
return (c); |
| } |
} |
| |
|
| int |
int |
| libnet_write_raw_ipv6(libnet_t *l, u_int8_t *packet, u_int32_t size) | libnet_write_raw_ipv6(libnet_t *l, const uint8_t *packet, uint32_t size) |
| { |
{ |
| |
int c = -1; |
| |
|
| #if defined HAVE_SOLARIS && !defined HAVE_SOLARIS_IPV6 |
#if defined HAVE_SOLARIS && !defined HAVE_SOLARIS_IPV6 |
| snprintf(l->err_buf, LIBNET_ERRBUF_SIZE, "%s(): no IPv6 support\n", | snprintf(l->err_buf, LIBNET_ERRBUF_SIZE, "%s(): no IPv6 support", |
| __func__, strerror(errno)); |
__func__, strerror(errno)); |
| return (-1); |
|
| } |
|
| #else |
#else |
| int c; |
|
| struct sockaddr_in6 sin; |
struct sockaddr_in6 sin; |
| struct libnet_ipv6_hdr *ip_hdr; |
struct libnet_ipv6_hdr *ip_hdr; |
| |
|
| if (l == NULL) |
if (l == NULL) |
| { | { |
| return (-1); |
return (-1); |
| } | } |
| |
|
| ip_hdr = (struct libnet_ipv6_hdr *)packet; |
ip_hdr = (struct libnet_ipv6_hdr *)packet; |
| |
|
|
Line 400 libnet_write_raw_ipv6(libnet_t *l, u_int8_t *packet, u
|
Line 330 libnet_write_raw_ipv6(libnet_t *l, u_int8_t *packet, u
|
| sin.sin6_family = AF_INET6; |
sin.sin6_family = AF_INET6; |
| memcpy(sin.sin6_addr.s6_addr, ip_hdr->ip_dst.libnet_s6_addr, |
memcpy(sin.sin6_addr.s6_addr, ip_hdr->ip_dst.libnet_s6_addr, |
| sizeof(ip_hdr->ip_dst.libnet_s6_addr)); |
sizeof(ip_hdr->ip_dst.libnet_s6_addr)); |
| | |
| c = sendto(l->fd, packet, size, 0, (struct sockaddr *)&sin, sizeof(sin)); |
c = sendto(l->fd, packet, size, 0, (struct sockaddr *)&sin, sizeof(sin)); |
| if (c != size) |
if (c != size) |
| { |
{ |
| #if !(__WIN32__) |
|
| snprintf(l->err_buf, LIBNET_ERRBUF_SIZE, |
snprintf(l->err_buf, LIBNET_ERRBUF_SIZE, |
| "%s(): %d bytes written (%s)\n", __func__, c, | "%s(): %d bytes written (%s)", __func__, c, |
| strerror(errno)); |
strerror(errno)); |
| #else /* __WIN32__ */ |
|
| snprintf(l->err_buf, LIBNET_ERRBUF_SIZE, |
|
| "%s(): %d bytes written (%d)\n", __func__, c, |
|
| WSAGetLastError()); |
|
| #endif /* !__WIN32__ */ |
|
| } |
} |
| |
#endif /* HAVE_SOLARIS && !HAVE_SOLARIS_IPV6 */ |
| return (c); |
return (c); |
| } |
} |
| #endif | #endif /* __WIN32__ */ |
| #endif | |
| /* EOF */ | /** |
| | * Local Variables: |
| | * indent-tabs-mode: nil |
| | * c-file-style: "stroustrup" |
| | * End: |
| | */ |