Skip to content

Commit

Permalink
Add getservbyname and getservbyport (#204)
Browse files Browse the repository at this point in the history
For every call of getservbyname/getservbyport the lookup of
/etc/services is done by opening the file and parsing each line
one-by-one. This is slow, but the implementation is simple.

This change also adds fixes for the gethostbyname function.
  • Loading branch information
ahgamut authored Jul 4, 2021
1 parent cfbd2af commit 0ea2907
Show file tree
Hide file tree
Showing 8 changed files with 412 additions and 25 deletions.
13 changes: 13 additions & 0 deletions libc/dns/ent.h
Original file line number Diff line number Diff line change
Expand Up @@ -16,12 +16,25 @@ struct hostent {
};
#define h_addr h_addr_list[0]

struct servent {
char *s_name; /* official service name */
char **s_aliases; /* alias list */
int s_port; /* port number (in network byte order) */
char *s_proto; /* protocol to use */
};

struct hostent *gethostent(void);
struct hostent *gethostbyname(const char *);
struct hostent *gethostbyaddr(const void *, socklen_t, int);
void sethostent(int);
void endhostent(void);

struct servent *getservent(void);
struct servent *getservbyname(const char *, const char *);
struct servent *getservbyport(int, const char *);
void setservent(int);
void endservent(void);

COSMOPOLITAN_C_END_
#endif /* !(__ASSEMBLER__ + __LINKER__ + 0) */
#endif /* COSMOPOLITAN_LIBC_DNS_ENT_H_ */
23 changes: 11 additions & 12 deletions libc/dns/gethostbyaddr.c
Original file line number Diff line number Diff line change
Expand Up @@ -31,23 +31,24 @@

struct hostent *gethostbyaddr(const void *s_addr, socklen_t len, int type) {
static struct hostent *ptr1, he1;
static char h_name[DNS_NAME_MAX+1];
static char* h_aliases[1];
static char* h_addr_list[2];
static char h_addr_list0[4];

struct sockaddr_in addr;
char name[DNS_NAME_MAX + 1];

if (!ptr1) {
he1.h_name = NULL;
he1.h_name = h_name;

he1.h_aliases = (char **)malloc(sizeof(char *) * 1);
if (!he1.h_aliases) return NULL;
he1.h_aliases = h_aliases;
he1.h_aliases[0] = NULL;

he1.h_addrtype = AF_INET;
he1.h_length = 4;
he1.h_addr_list = (char **)malloc(sizeof(char *) * 2);
if (!he1.h_addr_list) return NULL;
he1.h_addr_list = h_addr_list;

he1.h_addr_list[0] = (char *)malloc(sizeof(uint32_t));
if (!he1.h_addr_list[0]) return NULL;
he1.h_addr_list[0] = h_addr_list0;
he1.h_addr_list[1] = NULL;

ptr1 = &he1;
Expand All @@ -58,12 +59,10 @@ struct hostent *gethostbyaddr(const void *s_addr, socklen_t len, int type) {
addr.sin_port = 0;
addr.sin_addr.s_addr = *(uint32_t *)(s_addr);

if (getnameinfo((struct sockaddr *)&addr, sizeof(addr), name, sizeof(name),
NULL, 0, 0))
if (getnameinfo((struct sockaddr *)&addr, sizeof(addr), ptr1->h_name,
DNS_NAME_MAX, NULL, 0, 0))
return NULL;

if (ptr1->h_name) free(ptr1->h_name);
ptr1->h_name = strdup(name);
*((uint32_t *)ptr1->h_addr_list[0]) = (addr.sin_addr.s_addr);

return ptr1;
Expand Down
28 changes: 15 additions & 13 deletions libc/dns/gethostbyname.c
Original file line number Diff line number Diff line change
Expand Up @@ -27,39 +27,41 @@
╚─────────────────────────────────────────────────────────────────────────────*/
#include "libc/dns/ent.h"
#include "libc/mem/mem.h"
#include "libc/str/str.h"
#include "libc/sysv/consts/af.h"

struct hostent *gethostbyname(const char *name) {
static struct hostent *ptr0, he0;
static char h_name[DNS_NAME_MAX + 1];
static char *h_aliases[1];
static char *h_addr_list[2];
static char h_addr_list0[4];
struct addrinfo *result = NULL;

if (!ptr0) {
he0.h_name = NULL;
he0.h_name = h_name;

he0.h_aliases = (char **)malloc(sizeof(char *) * 1);
if (!he0.h_aliases) return NULL;
he0.h_aliases = h_aliases;
he0.h_aliases[0] = NULL;

he0.h_addrtype = AF_INET;
he0.h_length = 4;
he0.h_addr_list = (char **)malloc(sizeof(char *) * 2);
if (!he0.h_addr_list) return NULL;
he0.h_addr_list = h_addr_list;

he0.h_addr_list[0] = (char *)malloc(sizeof(uint32_t));
if (!he0.h_addr_list[0]) return NULL;
he0.h_addr_list[0] = h_addr_list0;
he0.h_addr_list[1] = NULL;

ptr0 = &he0;
}

if (getaddrinfo(name, NULL, NULL, &result) || result == NULL) return NULL;

if (ptr0->h_name) free(ptr0->h_name);
if (result->ai_canonname) {
ptr0->h_name = strdup(result->ai_canonname);
} else {
ptr0->h_name = strdup(name);
}
/* if getaddrinfo is successful, result->ai_canonname is non-NULL,
* (see newaddrinfo) but the string can still be empty */
if (result->ai_canonname[0])
memccpy(ptr0->h_name, result->ai_canonname, '\0', DNS_NAME_MAX);
else
memccpy(ptr0->h_name, name, '\0', DNS_NAME_MAX);

*((uint32_t *)ptr0->h_addr_list[0]) = (result->ai_addr4->sin_addr.s_addr);
/* TODO: if result has ai_next, fit multiple entries for h_addr_list */
Expand Down
65 changes: 65 additions & 0 deletions libc/dns/getservbyname.c
Original file line number Diff line number Diff line change
@@ -0,0 +1,65 @@
/*-*- mode:c;indent-tabs-mode:nil;c-basic-offset:2;tab-width:8;coding:utf-8 -*-│
│vi: set net ft=c ts=2 sts=2 sw=2 fenc=utf-8 :vi│
╞══════════════════════════════════════════════════════════════════════════════╡
│ This is free and unencumbered software released into the public domain. │
│ │
│ Anyone is free to copy, modify, publish, use, compile, sell, or │
│ distribute this software, either in source code form or as a compiled │
│ binary, for any purpose, commercial or non-commercial, and by any │
│ means. │
│ │
│ In jurisdictions that recognize copyright laws, the author or authors │
│ of this software dedicate any and all copyright interest in the │
│ software to the public domain. We make this dedication for the benefit │
│ of the public at large and to the detriment of our heirs and │
│ successors. We intend this dedication to be an overt act of │
│ relinquishment in perpetuity of all present and future rights to this │
│ software under copyright law. │
│ │
│ THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, │
│ EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF │
│ MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. │
│ IN NO EVENT SHALL THE AUTHORS BE LIABLE FOR ANY CLAIM, DAMAGES OR │
│ OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, │
│ ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR │
│ OTHER DEALINGS IN THE SOFTWARE. │
│ │
╚─────────────────────────────────────────────────────────────────────────────*/
#include "libc/dns/ent.h"
#include "libc/dns/servicestxt.h"
#include "libc/mem/mem.h"

struct servent *getservbyname(const char *name, const char *proto) {
static struct servent *ptr0, se0;
char *localproto = proto;
int p;

if (!ptr0) {
se0.s_name = NULL;
se0.s_aliases = (char **)malloc(sizeof(char *) * 1);
if (!se0.s_aliases) return NULL;
se0.s_aliases[0] = NULL;

se0.s_port = 0;
se0.s_proto = NULL;
ptr0 = &se0;
}

p = LookupServicesByName(name, &localproto);
if (p == -1) {
// localproto got alloc'd during the lookup?
if (!proto && localproto != proto) free(localproto);
return NULL;
}

ptr0->s_port = p;
if (ptr0->s_name) free(ptr0->s_name);
ptr0->s_name = strdup(name);

if (ptr0->s_proto) free(ptr0->s_proto);
ptr0->s_proto = strdup(localproto);

if (!proto && localproto != proto) free(localproto);

return ptr0;
}
63 changes: 63 additions & 0 deletions libc/dns/getservbyport.c
Original file line number Diff line number Diff line change
@@ -0,0 +1,63 @@
/*-*- mode:c;indent-tabs-mode:nil;c-basic-offset:2;tab-width:8;coding:utf-8 -*-│
│vi: set net ft=c ts=2 sts=2 sw=2 fenc=utf-8 :vi│
╞══════════════════════════════════════════════════════════════════════════════╡
│ This is free and unencumbered software released into the public domain. │
│ │
│ Anyone is free to copy, modify, publish, use, compile, sell, or │
│ distribute this software, either in source code form or as a compiled │
│ binary, for any purpose, commercial or non-commercial, and by any │
│ means. │
│ │
│ In jurisdictions that recognize copyright laws, the author or authors │
│ of this software dedicate any and all copyright interest in the │
│ software to the public domain. We make this dedication for the benefit │
│ of the public at large and to the detriment of our heirs and │
│ successors. We intend this dedication to be an overt act of │
│ relinquishment in perpetuity of all present and future rights to this │
│ software under copyright law. │
│ │
│ THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, │
│ EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF │
│ MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. │
│ IN NO EVENT SHALL THE AUTHORS BE LIABLE FOR ANY CLAIM, DAMAGES OR │
│ OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, │
│ ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR │
│ OTHER DEALINGS IN THE SOFTWARE. │
│ │
╚─────────────────────────────────────────────────────────────────────────────*/
#include "libc/dns/ent.h"
#include "libc/dns/servicestxt.h"

struct servent *getservbyport(int port, const char *proto) {
static struct servent *ptr1, se1;
char name[DNS_NAME_MAX];
char *localproto = proto;

if (!ptr1) {
se1.s_name = NULL;
se1.s_aliases = (char **)malloc(sizeof(char *) * 1);
if (!se1.s_aliases) return NULL;
se1.s_aliases[0] = NULL;

se1.s_port = 0;
se1.s_proto = NULL;
ptr1 = &se1;
}

if (LookupServicesByPort(port, &localproto, name, sizeof(name)) == -1) {
// localproto got alloc'd during the lookup?
if (!proto && localproto != proto) free(localproto);
return NULL;
}

ptr1->s_port = port;
if (ptr1->s_name) free(ptr1->s_name);
ptr1->s_name = strdup(name);

if (ptr1->s_proto) free(ptr1->s_proto);
ptr1->s_proto = strdup(localproto);

if (!proto && localproto != proto) free(localproto);

return ptr1;
}
38 changes: 38 additions & 0 deletions libc/dns/servent.c
Original file line number Diff line number Diff line change
@@ -0,0 +1,38 @@
/*-*- mode:c;indent-tabs-mode:nil;c-basic-offset:2;tab-width:8;coding:utf-8 -*-│
│vi: set net ft=c ts=2 sts=2 sw=2 fenc=utf-8 :vi│
╞══════════════════════════════════════════════════════════════════════════════╡
│ This is free and unencumbered software released into the public domain. │
│ │
│ Anyone is free to copy, modify, publish, use, compile, sell, or │
│ distribute this software, either in source code form or as a compiled │
│ binary, for any purpose, commercial or non-commercial, and by any │
│ means. │
│ │
│ In jurisdictions that recognize copyright laws, the author or authors │
│ of this software dedicate any and all copyright interest in the │
│ software to the public domain. We make this dedication for the benefit │
│ of the public at large and to the detriment of our heirs and │
│ successors. We intend this dedication to be an overt act of │
│ relinquishment in perpetuity of all present and future rights to this │
│ software under copyright law. │
│ │
│ THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, │
│ EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF │
│ MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. │
│ IN NO EVENT SHALL THE AUTHORS BE LIABLE FOR ANY CLAIM, DAMAGES OR │
│ OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, │
│ ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR │
│ OTHER DEALINGS IN THE SOFTWARE. │
│ │
╚─────────────────────────────────────────────────────────────────────────────*/
#include "libc/dns/ent.h"

struct servent *getservent(void) {
return NULL;
}

void setservent(int stayopen) {
}

void endservent(void) {
}
Loading

0 comments on commit 0ea2907

Please sign in to comment.