/* docs.idris-lang.org/en/latest/st/examples.html beej.us/guide/bgnet/html/ Sockets are non-standard, a simple Finite State Machine. The functions' only args are the socket. Socket functions requiring more parameters should be defined at the protocol level. The state machine exists to allow protocol code to execute code in the appropriate order. When calling a socket function, code can use the modified/unmodified states to determine next procedure. I included some code for IPV6, currently unused. */ #define SOCKET_STATE_READY 0 #define SOCKET_STATE_BIND_REQ 1 #define SOCKET_STATE_CONNECT_REQ 2 #define SOCKET_STATE_BOUND 3 #define SOCKET_STATE_LISTEN_REQ 4 #define SOCKET_STATE_LISTENING 5 #define SOCKET_STATE_OPEN 6 #define SOCKET_STATE_CLOSE_REQ 7 #define SOCKET_STATE_CLOSED 8 #define SOCKET_STREAM 1 #define SOCKET_DATAGRAM 2 #define SOCKET_RAW 3 #define AF_UNSPEC 0 #define AF_INET 2 #define AF_INET6 10 #define INADDR_ANY 0 #define INET_ADDRSTRLEN 16 //pubs.opengroup.com netinit/in.h #define INET6_ADDRSTRLEN 46 #define INET_MIN_ADDRSTRLEN 7 // ex: len of 0.0.0.0 #define INET6_MIN_ADDRSTRLEN 2 // ie: len of :: #define IP_PARSE_STATE_NUM 0 #define IP_PARSE_STATE_DOT 1 class CIPV4Address { U32 address; // 'in Network Byte order' ... Big Endian }; class CIPV6Address { U8 address[16]; // a clear #define would be nice }; class CIPAddressStorage {// class specifically meant to be generic casted either IPV4 or IPV6 Address. U8 padding[16]; }; class CSocketAddressIPV4 { U16 family; // 'AF_INET' U16 port; // 'in Network Byte order' ... Big Endian CIPV4Address address; U8 zeroes[8]; // 'same size as socket address' }; class CSocketAddressIPV6 { U16 family; // 'AF_INET6' U16 port; // 'in Network Byte order'... Big Endian U32 flow_info; CIPV6Address address; U32 scope_id; }; class CSocketAddressStorage {/* 'designed to be large enough to hold both IPV4 and IPV6 structures.' */ U16 family; U8 padding[26]; }; class CAddressInfo { I32 flags; I32 family; I32 socket_type; I32 protocol; I64 address_length; CSocketAddressStorage *address; U8 *canonical_name; CAddressInfo *next; }; class CSocket { U8 state; U16 type; U16 domain; }; U0 AddressInfoCopy(CAddressInfo *out, CAddressInfo *in) { // assumes *out already exists MemCopy(out, in, sizeof(CAddressInfo)); if (in->address) { out->address = CAlloc(in->address_length); MemCopy(out->address, in->address, in->address_length); } if (in->canonical_name) { out->canonical_name = StrNew(in->canonical_name); } } U0 AddressInfoFree(CAddressInfo *info) { CAddressInfo *next; while (info) { next = info->next; Free(info->address); Free(info->canonical_name); Free(info); info = next; } } Bool IPV4AddressParse(U8 *string, U32 *destination) { U8 *lexable_string = StrNew(string); lexable_string = StrReplace(lexable_string, ".", ","); // swap dots with commas since Lex is easier with them. CCompCtrl *cc = CompCtrlNew(lexable_string); //Bts(&cc->opts, OPTf_DECIMAL_ONLY); cc->opts |= 1 << OPTf_DECIMAL_ONLY; I64 tk; I64 state = IP_PARSE_STATE_NUM; U32 temp_destination = 0; I64 current_section = 0; // IPV4 address has 4 total sections while (tk = Lex(cc)) { switch (state) { case IP_PARSE_STATE_NUM: switch (tk) { case TK_I64: if (cc->cur_i64 > 255 || cc->cur_i64 < 0) { NetErr("IPV4 ADDRESS PARSE: Invalid value, must be 0 - 255."); return FALSE; } if (current_section > 3) { NetErr("IPV4 ADDRESS PARSE: IP Address can only have 4 sections."); return FALSE; } temp_destination |= cc->cur_i64 << (current_section * 8); current_section++; state = IP_PARSE_STATE_DOT; break; default: NetErr("IPV4 ADDRESS PARSE: Expected decimal. "); return FALSE; } break; case IP_PARSE_STATE_DOT: switch (tk) { case ',': state = IP_PARSE_STATE_NUM; break; default: NetErr("IPV4 ADDRESS PARSE: Expected dot. "); return FALSE; } break; } } CompCtrlDel(cc); temp_destination = EndianU32(temp_destination); // store the address in Network Byte Order (Big-Endian) *destination = temp_destination; return TRUE; } I64 PresentationToNetwork(I64 address_family, U8 *string, CIPAddressStorage *destination) { /* Converts IP string to internet address class, our inet_pton(). Destination written as CIPV4Address or CIPV6Address depending on value of address_family. The destination address is the generic class, functions calling this method must cast their classes in the params. */ CIPV4Address *ipv4_address; CIPV6Address *ipv6_address; I64 string_length = StrLen(string); switch (address_family) { case AF_INET: if (string_length > INET_ADDRSTRLEN || string_length < INET_MIN_ADDRSTRLEN) { NetErr("IP to Socket Address failed: Invalid Input String Size."); return -1; } ipv4_address = destination; if (!IPV4AddressParse(string, &ipv4_address->address)) return -1; break; case AF_INET6: if (string_length > INET6_ADDRSTRLEN || string_length < INET6_MIN_ADDRSTRLEN) { NetErr("IP to Socket Address failed: Invalid Input String Size."); return -1; } ipv6_address = destination; NetErr("IP to Socket Address failed: FIXME, IPV6 support not implemented yet.\n"); return -1; default: NetErr("IP to Socket Address failed: Invalid Address Family."); return -1; } return 0; } U8 *NetworkToPresentation(I64 address_family, CIPAddressStorage *source) { // converts socket address to IP string, our inet_ntop. Taking Shrine approach of function returns U8* . U8 *ip_string = NULL; CIPV4Address *ipv4_source; CIPV4Address *ipv6_source; switch (address_family) { case AF_INET: ipv4_source = source; ip_string = MStrPrint("%d.%d.%d.%d", ipv4_source->address.u8[0], ipv4_source->address.u8[1], ipv4_source->address.u8[2], ipv4_source->address.u8[3]); break; case AF_INET6: ipv6_source = source; NetErr("Socket Address to IP failed: FIXME, IPV6 support not implemented yet.\n"); return -1; break; default: NetErr("Socket Address to IP failed: Invalid Address Family."); break; } return ip_string; } CSocket *Socket(U16 domain, U16 type) { CSocket *socket = CAlloc(sizeof(CSocket)); socket->domain = domain; socket->type = type; socket->state = SOCKET_STATE_READY; return socket; } U0 SocketStateErr(U8 *request, U8 state) { U8 *state_string; switch (state) { case SOCKET_STATE_READY: state_string = StrNew("READY"); break; case SOCKET_STATE_BIND_REQ: state_string = StrNew("BIND REQUEST"); break; case SOCKET_STATE_CONNECT_REQ: state_string = StrNew("CONNECT REQUEST"); break; case SOCKET_STATE_BOUND: state_string = StrNew("BOUND"); break; case SOCKET_STATE_LISTEN_REQ: state_string = StrNew("LISTEN REQUEST"); break; case SOCKET_STATE_LISTENING: state_string = StrNew("LISTENING"); break; case SOCKET_STATE_OPEN: state_string = StrNew("OPEN"); break; case SOCKET_STATE_CLOSE_REQ: state_string = StrNew("CLOSE REQUEST"); break; case SOCKET_STATE_CLOSED: state_string = StrNew("CLOSED"); break; } NetErr("Socket attempted %s while in %s state.", request, state_string); } Bool SocketAccept(CSocket *socket) { switch (socket->state) { case SOCKET_STATE_LISTENING: /* Socket expected to stay listening. At protocol level, a new socket 'connected' to this one is expected to be made. */ return TRUE; default: SocketStateErr("ACCEPT", socket->state); return FALSE; } } Bool SocketClose(CSocket *socket) { /* Sockets attempting close will enter close request state before destroying the socket at the protocol level. */ socket->state = SOCKET_STATE_CLOSE_REQ; return TRUE; } Bool SocketBind(CSocket *socket) { switch (socket->state) { case SOCKET_STATE_READY: /* Sockets can only be bound if they are in initial state. */ socket->state = SOCKET_STATE_BIND_REQ; return TRUE; default: SocketStateErr("BIND", socket->state); return FALSE; } } Bool SocketConnect(CSocket *socket) { switch (socket->state) { case SOCKET_STATE_READY: /* Sockets can only be connected if they are in initial state. */ socket->state = SOCKET_STATE_CONNECT_REQ; return TRUE; default: SocketStateErr("CONNECT", socket->state); return FALSE; } } Bool SocketListen(CSocket *socket) { switch (socket->state) { case SOCKET_STATE_BOUND: /* A socket must be bound to set it to listening. */ socket->state = SOCKET_STATE_LISTEN_REQ; return TRUE; default: SocketStateErr("LISTEN", socket->state); return FALSE; } } Bool SocketReceive(CSocket *socket) { switch (socket->state) { case SOCKET_STATE_OPEN: case SOCKET_STATE_BOUND: /* Sockets can only recv when connected to or bound. */ return TRUE; default: SocketStateErr("RECEIVE", socket->state); return FALSE; } } Bool SocketReceiveFrom(CSocket *socket) { switch (socket->state) { case SOCKET_STATE_OPEN: case SOCKET_STATE_BOUND: /* Sockets can only recvfrom when connected to or bound. */ return TRUE; default: SocketStateErr("RECEIVE FROM", socket->state); return FALSE; } } Bool SocketSend(CSocket *socket) { switch (socket->state) { case SOCKET_STATE_OPEN: /* Sockets can only send when they have been connected to. */ return TRUE; default: SocketStateErr("SEND", socket->state); return FALSE; } } Bool SocketSendTo(CSocket *socket) { switch (socket->state) { case SOCKET_STATE_OPEN: case SOCKET_STATE_BOUND: case SOCKET_STATE_READY: /* Sockets can only sendto when connected to, bound, or in initial state. Protocol logic will determine how to change state based on params. */ return TRUE; default: SocketStateErr("SEND TO", socket->state); return FALSE; } }