1
0

getservbyport_r.c 1.2 KB

12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455
  1. #define _GNU_SOURCE
  2. #include <sys/socket.h>
  3. #include <netinet/in.h>
  4. #include <netdb.h>
  5. #include <inttypes.h>
  6. #include <errno.h>
  7. #include <string.h>
  8. int getservbyport_r(int port, const char *prots,
  9. struct servent *se, char *buf, size_t buflen, struct servent **res)
  10. {
  11. int i;
  12. struct sockaddr_in sin = {
  13. .sin_family = AF_INET,
  14. .sin_port = port,
  15. };
  16. if (!prots) {
  17. int r = getservbyport_r(port, "tcp", se, buf, buflen, res);
  18. if (r) r = getservbyport_r(port, "udp", se, buf, buflen, res);
  19. return r;
  20. }
  21. /* Align buffer */
  22. i = (uintptr_t)buf & sizeof(char *)-1;
  23. if (!i) i = sizeof(char *);
  24. if (buflen < 3*sizeof(char *)-i)
  25. return ERANGE;
  26. buf += sizeof(char *)-i;
  27. buflen -= sizeof(char *)-i;
  28. if (strcmp(prots, "tcp") && strcmp(prots, "udp")) return EINVAL;
  29. se->s_port = port;
  30. se->s_proto = (char *)prots;
  31. se->s_aliases = (void *)buf;
  32. buf += 2*sizeof(char *);
  33. buflen -= 2*sizeof(char *);
  34. se->s_aliases[1] = 0;
  35. se->s_aliases[0] = se->s_name = buf;
  36. switch (getnameinfo((void *)&sin, sizeof sin, 0, 0, buf, buflen,
  37. strcmp(prots, "udp") ? 0 : NI_DGRAM)) {
  38. case EAI_MEMORY:
  39. case EAI_SYSTEM:
  40. return ENOMEM;
  41. default:
  42. return ENOENT;
  43. case 0:
  44. break;
  45. }
  46. *res = se;
  47. return 0;
  48. }