Error compiling code using winsock functions

[Date Prev][Date Next][Thread Prev][Thread Next][Date Index][Thread Index]

 



Hi there, I've recently been trying some programming with sockets in windows, and I obtained the following code from a microsoft book called "Network programming for microsoft windows"...
the code is as follows:


// Module Name: iphdrinc.c
//
#pragma pack(1)

#define WIN32_LEAN_AND_MEAN

#include <winsock2.h>
#include <ws2tcpip.h>

#include <stdio.h>
#include <stdlib.h>

#define MAX_MESSAGE        4068
#define MAX_PACKET         4096
//
// Set up some default values
//
#define DEFAULT_PORT       5150
#define DEFAULT_IP         "10.0.0.1"
#define DEFAULT_COUNT      5
#define DEFAULT_MESSAGE    "This is a test"

//
// Define the IP header. Make the version and length fields one
// character since we can't declare two 4-bit fields without
// the compiler aligning them on at least a 1-byte boundary.
//
typedef struct ip_hdr
{
   unsigned char  ip_verlen;        // IP version & length
   unsigned char  ip_tos;           // IP type of service
   unsigned short ip_totallength;   // Total length
   unsigned short ip_id;            // Unique identifier
   unsigned short ip_offset;        // Fragment offset field
   unsigned char  ip_ttl;           // Time to live
   unsigned char  ip_protocol;      // Protocol(TCP, UDP, etc.)
   unsigned short ip_checksum;      // IP checksum
   unsigned int   ip_srcaddr;       // Source address
   unsigned int   ip_destaddr;      // Destination address
} IP_HDR, *PIP_HDR, FAR* LPIP_HDR;
//
// Define the UDP header
//
typedef struct udp_hdr
{
   unsigned short src_portno;       // Source port number
   unsigned short dst_portno;       // Destination port number
   unsigned short udp_length;       // UDP packet length
   unsigned short udp_checksum;     // UDP checksum (optional)
} UDP_HDR, *PUDP_HDR;

//
// Global variables
//
unsigned long  dwToIP,               // IP to send to
              dwFromIP;             // IP to send from (spoof)
unsigned short iToPort,              // Port to send to
              iFromPort;            // Port to send from (spoof)
DWORD          dwCount;              // Number of times to send
char           strMessage[MAX_MESSAGE]; // Message to send

//
// Description:
//    Print usage information and exit
//
void usage(char *progname)
{
   printf("usage: %s [-fp:int] [-fi:str] [-tp:int] [-ti:str]\
       [-n:int] [-m:str]\n", progname);
   printf("       -fp:int   From (sender) port number\n");
   printf("       -fi:IP    From (sender) IP address\n");
   printf("       -fp:int   To (recipient) port number\n");
   printf("       -fi:IP    To (recipient) IP address\n");
   printf("       -n:int    Number of times to read message\n");
   printf("       -m:str    Size of buffer to read\n\n");
   ExitProcess(1);
}
//
// Function: ValidateArgs
//
// Description:
//    Parse the command line arguments, and set some global flags to
//    indicate the actions to perform
//
void ValidateArgs(int argc, char **argv)
{
   int                i;

   iToPort = DEFAULT_PORT;
   iFromPort = DEFAULT_PORT;
   dwToIP = inet_addr(DEFAULT_IP);
   dwFromIP = inet_addr(DEFAULT_IP);
   dwCount = DEFAULT_COUNT;
   strcpy(strMessage, DEFAULT_MESSAGE);

   for(i = 1; i < argc; i++)
   {
       if ((argv[i][0] == '-') || (argv[i][0] == '/'))
       {
           switch (tolower(argv[i][1]))
           {
               case 'f':        // From address
                   switch (tolower(argv[i][2]))
                   {
                       case 'p':
                           if (strlen(argv[i]) > 4)
                               iFromPort = atoi(&argv[i][4]);
                           break;
                       case 'i':
                           if (strlen(argv[i]) > 4)
                               dwFromIP = inet_addr(&argv[i][4]);
                           break;
                       default:
                           usage(argv[0]);
                           break;
                   }
                   break;
               case 't':        // To address
                   switch (tolower(argv[i][2]))
                   {
                       case 'p':
                           if (strlen(argv[i]) > 4)
                               iToPort = atoi(&argv[i][4]);
                           break;
                       case 'i':
                           if (strlen(argv[i]) > 4)
                               dwToIP = inet_addr(&argv[i][4]);
                           break;
                       default:
                           usage(argv[0]);
                           break;
                   }
                   break;
               case 'n':        // Number of times to send message
                   if (strlen(argv[i]) > 3)
                       dwCount = atol(&argv[i][3]);
                   break;
               case 'm':
                   if (strlen(argv[i]) > 3)
                       strcpy(strMessage, &argv[i][3]);
                   break;
               default:
                   usage(argv[0]);
                   break;
           }
       }
   }
   return;
}

//
// Function: checksum
//
// Description:
//    This function calculates the 16-bit one's complement sum
//    for the supplied buffer
//
USHORT checksum(USHORT *buffer, int size)
{
   unsigned long cksum=0;

   while (size > 1)
   {
       cksum += *buffer++;
       size  -= sizeof(USHORT);
   }
   if (size)
   {
       cksum += *(UCHAR*)buffer;
   }
   cksum = (cksum >> 16) + (cksum & 0xffff);
   cksum += (cksum >>16);

   return (USHORT)(~cksum);
}

//
// Function: main
//
// Description:
//    First parse command line arguments and load Winsock. Then
//    create the raw socket and set the IP_HDRINCL option.
//    Following this, assemble the IP and UDP packet headers by
//    assigning the correct values and calculating the checksums.
//    Then fill in the data and send to its destination.
//
int main(int argc, char **argv)
{
   WSADATA            wsd;
   SOCKET             s;
   BOOL               bOpt;
   struct sockaddr_in remote;       // IP addressing structures
   IP_HDR             ipHdr;
   UDP_HDR            udpHdr;
   int                ret;
   DWORD              i;
   unsigned short     iTotalSize,   // Lots of sizes needed to fill
                      iUdpSize,     // the various headers with
                      iUdpChecksumSize,
                      iIPVersion,
                      iIPSize,
                      cksum = 0;
   char               buf[MAX_PACKET],
                     *ptr = NULL;
   IN_ADDR            addr;

   // Parse command line arguments, and print them out
   //
   ValidateArgs(argc, argv);
   addr.S_un.S_addr = dwFromIP;
   printf("From IP: <%s>\n     Port: %d\n", inet_ntoa(addr),
       iFromPort);
   addr.S_un.S_addr = dwToIP;
   printf("To   IP: <%s>\n     Port: %d\n", inet_ntoa(addr),
       iToPort);
   printf("Message: [%s]\n", strMessage);
   printf("Count:   %d\n", dwCount);

   if (WSAStartup(MAKEWORD(2,2), &wsd) != 0)
   {
       printf("WSAStartup() failed: %d\n", GetLastError());
       return -1;
   }
   //  Creating a raw socket
   //
   s = WSASocket(AF_INET, SOCK_RAW, IPPROTO_UDP, NULL, 0,0);
   if (s == INVALID_SOCKET)
   {
       printf("WSASocket() failed: %d\n", WSAGetLastError());
       return -1;
   }

   // Enable the IP header include option
   //
   bOpt = TRUE;
   ret = setsockopt(s, IPPROTO_IP, IP_HDRINCL, (char *)&bOpt,
      sizeof(bOpt));
   if (ret == SOCKET_ERROR)
   {
       printf("setsockopt(IP_HDRINCL) failed: %d\n", WSAGetLastError());
       return -1;
   }
   // Initalize the IP header
   //
   iTotalSize = sizeof(ipHdr) + sizeof(udpHdr) + strlen(strMessage);

   iIPVersion = 4;
   iIPSize = sizeof(ipHdr) / sizeof(unsigned long);
   //
   // IP version goes in the high-order 4 bits of ip_verlen. The
   // IP header length (in 32-bit words) goes in the lower 4 bits.
   //
   ipHdr.ip_verlen = (iIPVersion << 4) | iIPSize;
   ipHdr.ip_tos = 0;                         // IP type of service
   ipHdr.ip_totallength = htons(iTotalSize); // Total packet len
   ipHdr.ip_id = 0;                 // Unique identifier: set to 0
   ipHdr.ip_offset = 0;             // Fragment offset field
   ipHdr.ip_ttl = 128;              // Time to live
   ipHdr.ip_protocol = 0x11;        // Protocol(UDP)
   ipHdr.ip_checksum = 0 ;          // IP checksum
   ipHdr.ip_srcaddr = dwFromIP;     // Source address
   ipHdr.ip_destaddr = dwToIP;      // Destination address
   //
   // Initalize the UDP header
   //
   iUdpSize = sizeof(udpHdr) + strlen(strMessage);

   udpHdr.src_portno = htons(iFromPort) ;
   udpHdr.dst_portno = htons(iToPort) ;
   udpHdr.udp_length = htons(iUdpSize) ;
   udpHdr.udp_checksum = 0 ;
   //
   // Build the UDP pseudo-header for calculating the UDP checksum.
   // The pseudo-header consists of the 32-bit source IP address,
   // the 32-bit destination IP address, a zero byte, the 8-bit
   // IP protocol field, the 16-bit UDP length, and the UDP
   // header itself along with its data (padded with a 0 if
   // the data is odd length).
   //
   iUdpChecksumSize = 0;
   ptr = buf;
   ZeroMemory(buf, MAX_PACKET);

   memcpy(ptr, &ipHdr.ip_srcaddr,  sizeof(ipHdr.ip_srcaddr));
   ptr += sizeof(ipHdr.ip_srcaddr);
   iUdpChecksumSize += sizeof(ipHdr.ip_srcaddr);

   memcpy(ptr, &ipHdr.ip_destaddr, sizeof(ipHdr.ip_destaddr));
   ptr += sizeof(ipHdr.ip_destaddr);
   iUdpChecksumSize += sizeof(ipHdr.ip_destaddr);

   ptr++;
   iUdpChecksumSize += 1;

   memcpy(ptr, &ipHdr.ip_protocol, sizeof(ipHdr.ip_protocol));
   ptr += sizeof(ipHdr.ip_protocol);
   iUdpChecksumSize += sizeof(ipHdr.ip_protocol);

   memcpy(ptr, &udpHdr.udp_length, sizeof(udpHdr.udp_length));
   ptr += sizeof(udpHdr.udp_length);
   iUdpChecksumSize += sizeof(udpHdr.udp_length);

   memcpy(ptr, &udpHdr, sizeof(udpHdr));
   ptr += sizeof(udpHdr);
   iUdpChecksumSize += sizeof(udpHdr);

   for(i = 0; i < strlen(strMessage); i++, ptr++)
       *ptr = strMessage[i];
   iUdpChecksumSize += strlen(strMessage);

   cksum = checksum((USHORT *)buf, iUdpChecksumSize);
   udpHdr.udp_checksum = cksum;
   //
   // Now assemble the IP and UDP headers along with the data
   // so we can send it
   //
   ZeroMemory(buf, MAX_PACKET);
   ptr = buf;

   memcpy(ptr, &ipHdr, sizeof(ipHdr));   ptr += sizeof(ipHdr);
   memcpy(ptr, &udpHdr, sizeof(udpHdr)); ptr += sizeof(udpHdr);
   memcpy(ptr, strMessage, strlen(strMessage));

   // Apparently, this SOCKADDR_IN structure makes no difference.
   // Whatever we put as the destination IP addr in the IP header
   // is what goes. Specifying a different destination in remote
   // will be ignored.
   //
   remote.sin_family = AF_INET;
   remote.sin_port = htons(iToPort);
   remote.sin_addr.s_addr = dwToIP;

   for(i = 0; i < dwCount; i++)
   {
       ret = sendto(s, buf, iTotalSize, 0, (SOCKADDR *)&remote,
           sizeof(remote));
       if (ret == SOCKET_ERROR)
       {
           printf("sendto() failed: %d\n", WSAGetLastError());
           break;
       }
       else
           printf("sent %d bytes\n", ret);
   }
   closesocket(s) ;
   WSACleanup() ;

   return 0;
}


------- END CODE -----------


Basically, when compiling with cygwin with the following command:
gcc iphdrinc.c -o iphdrinc.exe -lwsock32

I get the following error:
"iphdrinc.o(.text+0x820):iphdrinc.c: undefined reference to '_WSASocketA@24' collect2: ld returned 1 exit status"


and if I dont use -lwsock32, i get even more errors of the same form ("undefined reference to XXX", where XXX is some function in winsock2.h)..
and I've noticed that the WSASocketA function is stated in the winsock2.h header file as follows:
SOCKET WINAPI WSASocketA(int, int, int, LPWSAPROTOCOL_INFOA, GROUP, DWORD);


(if that helps)

anyways, Im running windows XP, and the latest version of cygwin... so hopefully someone can give me some help =)

thanks in advance !
regards,
David.

_________________________________________________________________
Protect your PC - get McAfee.com VirusScan Online http://clinic.mcafee.com/clinic/ibuy/campaign.asp?cid=3963



[Index of Archives]     [Linux C Programming]     [Linux Kernel]     [eCos]     [Fedora Development]     [Fedora Announce]     [Autoconf]     [The DWARVES Debugging Tools]     [Yosemite Campsites]     [Yosemite News]     [Linux GCC]

  Powered by Linux