2021-07-04 23:31:42 +00:00
|
|
|
/*-*- 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/servicestxt.h"
|
|
|
|
|
|
|
|
#include "libc/bits/safemacros.internal.h"
|
|
|
|
#include "libc/dce.h"
|
|
|
|
#include "libc/errno.h"
|
|
|
|
#include "libc/fmt/conv.h"
|
|
|
|
#include "libc/fmt/fmt.h"
|
|
|
|
#include "libc/macros.internal.h"
|
|
|
|
#include "libc/mem/mem.h"
|
|
|
|
#include "libc/nt/systeminfo.h"
|
|
|
|
#include "libc/runtime/runtime.h"
|
|
|
|
#include "libc/str/str.h"
|
|
|
|
|
|
|
|
static textwindows noinline char *GetNtServicesTxtPath(char *pathbuf,
|
|
|
|
uint32_t size) {
|
|
|
|
const char *const kWinHostsPath = "\\drivers\\etc\\services";
|
|
|
|
uint32_t len = GetSystemDirectoryA(&pathbuf[0], size);
|
|
|
|
if (len && len + strlen(kWinHostsPath) + 1 < size) {
|
|
|
|
if (pathbuf[len] == '\\') pathbuf[len--] = '\0';
|
|
|
|
memcpy(&pathbuf[len], kWinHostsPath, strlen(kWinHostsPath) + 1);
|
|
|
|
return &pathbuf[0];
|
|
|
|
} else {
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Opens and searches /etc/services to find name for a given port.
|
|
|
|
*
|
|
|
|
* format of /etc/services is like this:
|
|
|
|
*
|
|
|
|
* # comment
|
|
|
|
* # NAME PORT/PROTOCOL ALIASES
|
|
|
|
*
|
|
|
|
* ftp 21/tcp
|
|
|
|
* fsp 21/udp fspd
|
|
|
|
* ssh 22/tcp
|
|
|
|
*
|
|
|
|
* @param servport is the port number (in network byte order)
|
|
|
|
* @param servproto is a pointer to a string (*servproto can be NULL)
|
2021-07-05 19:25:26 +00:00
|
|
|
* @param buf is a buffer to store the official name of the service
|
2021-07-04 23:31:42 +00:00
|
|
|
* @param bufsize is the size of buf
|
2021-07-05 19:25:26 +00:00
|
|
|
* @param filepath is the location of the services file
|
|
|
|
* (if NULL, uses /etc/services)
|
2021-07-04 23:31:42 +00:00
|
|
|
* @returns 0 on success, -1 on error
|
|
|
|
*
|
|
|
|
* @note aliases are not read from the file.
|
|
|
|
*/
|
|
|
|
int LookupServicesByPort(const int servport, char **servproto, char *buf,
|
2021-07-05 19:25:26 +00:00
|
|
|
size_t bufsize, const char *filepath) {
|
2021-07-04 23:31:42 +00:00
|
|
|
FILE *f;
|
|
|
|
char *line;
|
|
|
|
char pathbuf[PATH_MAX];
|
|
|
|
const char *path;
|
|
|
|
size_t linesize;
|
2021-07-05 19:25:26 +00:00
|
|
|
int found;
|
2021-07-04 23:31:42 +00:00
|
|
|
char *name, *port, *proto, *comment, *tok;
|
|
|
|
|
2021-07-05 19:25:26 +00:00
|
|
|
if (!(path = filepath)) {
|
|
|
|
path = "/etc/services";
|
|
|
|
if (IsWindows()) {
|
|
|
|
path =
|
|
|
|
firstnonnull(GetNtServicesTxtPath(pathbuf, ARRAYLEN(pathbuf)), path);
|
|
|
|
}
|
2021-07-04 23:31:42 +00:00
|
|
|
}
|
2021-07-05 19:25:26 +00:00
|
|
|
|
2021-07-04 23:31:42 +00:00
|
|
|
if (bufsize == 0 || !(f = fopen(path, "r"))) {
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
line = NULL;
|
|
|
|
linesize = 0;
|
|
|
|
found = 0;
|
|
|
|
|
|
|
|
while (found == 0 && (getline(&line, &linesize, f)) != -1) {
|
|
|
|
if ((comment = strchr(line, '#'))) *comment = '\0';
|
|
|
|
name = strtok_r(line, " \t\r\n\v", &tok);
|
|
|
|
port = strtok_r(NULL, "/ \t\r\n\v", &tok);
|
2021-07-05 19:25:26 +00:00
|
|
|
proto = strtok_r(NULL, " \t\r\n\v", &tok);
|
|
|
|
if (name && port && proto && servport == htons(atoi(port))) {
|
|
|
|
if (!servproto[0]) {
|
2021-07-04 23:31:42 +00:00
|
|
|
servproto[0] = strdup(proto);
|
|
|
|
strncpy(buf, name, bufsize);
|
|
|
|
found = 1;
|
2021-07-05 19:25:26 +00:00
|
|
|
} else if (strcasecmp(proto, servproto[0]) == 0) {
|
2021-07-04 23:31:42 +00:00
|
|
|
strncpy(buf, name, bufsize);
|
|
|
|
found = 1;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
free(line);
|
|
|
|
|
|
|
|
if (ferror(f)) {
|
|
|
|
errno = ferror(f);
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
fclose(f);
|
|
|
|
|
|
|
|
if (!found) return -1;
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Opens and searches /etc/services to find port for a given name.
|
|
|
|
*
|
|
|
|
* @param servname is a NULL-terminated string
|
|
|
|
* @param servproto is a pointer to a string (*servproto can be NULL)
|
2021-07-05 19:25:26 +00:00
|
|
|
* @param buf is a buffer to store the official name of the service
|
|
|
|
* @param bufsize is the size of buf
|
|
|
|
* @param filepath is the location of services file
|
|
|
|
* (if NULL, uses /etc/services)
|
2021-07-04 23:31:42 +00:00
|
|
|
* @returns -1 on error, or
|
|
|
|
* positive port number (in network byte order)
|
|
|
|
*
|
2021-07-05 19:25:26 +00:00
|
|
|
* @note aliases are read from file for comparison, but not returned.
|
2021-07-04 23:31:42 +00:00
|
|
|
* @see LookupServicesByPort
|
|
|
|
*/
|
2021-07-05 19:25:26 +00:00
|
|
|
int LookupServicesByName(const char *servname, char **servproto, char *buf,
|
|
|
|
size_t bufsize, const char *filepath) {
|
2021-07-04 23:31:42 +00:00
|
|
|
FILE *f;
|
|
|
|
char *line;
|
|
|
|
char pathbuf[PATH_MAX];
|
|
|
|
const char *path;
|
|
|
|
size_t linesize;
|
2021-07-05 19:25:26 +00:00
|
|
|
int found, result;
|
|
|
|
char *name, *port, *proto, *alias, *comment, *tok;
|
2021-07-04 23:31:42 +00:00
|
|
|
|
2021-07-05 19:25:26 +00:00
|
|
|
if (!(path = filepath)) {
|
|
|
|
path = "/etc/services";
|
|
|
|
if (IsWindows()) {
|
|
|
|
path =
|
|
|
|
firstnonnull(GetNtServicesTxtPath(pathbuf, ARRAYLEN(pathbuf)), path);
|
|
|
|
}
|
2021-07-04 23:31:42 +00:00
|
|
|
}
|
2021-07-05 19:25:26 +00:00
|
|
|
|
|
|
|
if (bufsize == 0 || !(f = fopen(path, "r"))) {
|
2021-07-04 23:31:42 +00:00
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
line = NULL;
|
|
|
|
linesize = 0;
|
|
|
|
found = 0;
|
|
|
|
result = -1;
|
|
|
|
|
|
|
|
while (found == 0 && (getline(&line, &linesize, f)) != -1) {
|
|
|
|
if ((comment = strchr(line, '#'))) *comment = '\0';
|
|
|
|
name = strtok_r(line, " \t\r\n\v", &tok);
|
|
|
|
port = strtok_r(NULL, "/ \t\r\n\v", &tok);
|
2021-07-05 19:25:26 +00:00
|
|
|
proto = strtok_r(NULL, " \t\r\n\v", &tok);
|
|
|
|
if (name && port && proto) {
|
|
|
|
alias = name;
|
|
|
|
while (alias && strcasecmp(alias, servname) != 0)
|
|
|
|
alias = strtok_r(NULL, " \t\r\n\v", &tok);
|
|
|
|
|
|
|
|
if (alias) /* alias matched with servname */
|
|
|
|
{
|
|
|
|
if (!servproto[0]) {
|
|
|
|
servproto[0] = strdup(proto);
|
|
|
|
result = htons(atoi(port));
|
|
|
|
strncpy(buf, name, bufsize);
|
|
|
|
found = 1;
|
|
|
|
} else if (strcasecmp(proto, servproto[0]) == 0) {
|
|
|
|
result = htons(atoi(port));
|
|
|
|
strncpy(buf, name, bufsize);
|
|
|
|
found = 1;
|
|
|
|
}
|
2021-07-04 23:31:42 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
free(line);
|
|
|
|
|
|
|
|
if (ferror(f)) {
|
|
|
|
errno = ferror(f);
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
fclose(f);
|
|
|
|
|
|
|
|
if (!found) return -1;
|
|
|
|
|
|
|
|
return result;
|
|
|
|
}
|