utils.c 3.7 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157
  1. /*
  2. * utils - misc libubox utility functions
  3. *
  4. * Copyright (C) 2012 Felix Fietkau <nbd@openwrt.org>
  5. *
  6. * Permission to use, copy, modify, and/or distribute this software for any
  7. * purpose with or without fee is hereby granted, provided that the above
  8. * copyright notice and this permission notice appear in all copies.
  9. *
  10. * THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL WARRANTIES
  11. * WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF
  12. * MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR
  13. * ANY SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES
  14. * WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN
  15. * ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF
  16. * OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE.
  17. */
  18. #include <sys/mman.h>
  19. #include <stdarg.h>
  20. #include <stdlib.h>
  21. #include <stdio.h>
  22. #include "utils.h"
  23. #define foreach_arg(_arg, _addr, _len, _first_addr, _first_len) \
  24. for (_addr = (_first_addr), _len = (_first_len); \
  25. _addr; \
  26. _addr = va_arg(_arg, void **), _len = _addr ? va_arg(_arg, size_t) : 0)
  27. #define C_PTR_ALIGN (sizeof(size_t))
  28. #define C_PTR_MASK (-C_PTR_ALIGN)
  29. void *__calloc_a(size_t len, ...)
  30. {
  31. va_list ap, ap1;
  32. void *ret;
  33. void **cur_addr;
  34. size_t cur_len;
  35. int alloc_len = 0;
  36. char *ptr;
  37. va_start(ap, len);
  38. va_copy(ap1, ap);
  39. foreach_arg(ap1, cur_addr, cur_len, &ret, len)
  40. alloc_len += (cur_len + C_PTR_ALIGN - 1 ) & C_PTR_MASK;
  41. va_end(ap1);
  42. ptr = calloc(1, alloc_len);
  43. if (!ptr) {
  44. va_end(ap);
  45. return NULL;
  46. }
  47. alloc_len = 0;
  48. foreach_arg(ap, cur_addr, cur_len, &ret, len) {
  49. *cur_addr = &ptr[alloc_len];
  50. alloc_len += (cur_len + C_PTR_ALIGN - 1) & C_PTR_MASK;
  51. }
  52. va_end(ap);
  53. return ret;
  54. }
  55. #ifdef LIBUBOX_COMPAT_CLOCK_GETTIME
  56. #include <mach/mach_host.h> /* host_get_clock_service() */
  57. #include <mach/mach_port.h> /* mach_port_deallocate() */
  58. #include <mach/mach_init.h> /* mach_host_self(), mach_task_self() */
  59. #include <mach/clock.h> /* clock_get_time() */
  60. static clock_serv_t clock_realtime;
  61. static clock_serv_t clock_monotonic;
  62. static void __constructor clock_name_init(void)
  63. {
  64. mach_port_t host_self = mach_host_self();
  65. host_get_clock_service(host_self, CLOCK_REALTIME, &clock_realtime);
  66. host_get_clock_service(host_self, CLOCK_MONOTONIC, &clock_monotonic);
  67. }
  68. static void __destructor clock_name_dealloc(void)
  69. {
  70. mach_port_t self = mach_task_self();
  71. mach_port_deallocate(self, clock_realtime);
  72. mach_port_deallocate(self, clock_monotonic);
  73. }
  74. int clock_gettime(int type, struct timespec *tv)
  75. {
  76. int retval = -1;
  77. mach_timespec_t mts;
  78. switch (type) {
  79. case CLOCK_REALTIME:
  80. retval = clock_get_time(clock_realtime, &mts);
  81. break;
  82. case CLOCK_MONOTONIC:
  83. retval = clock_get_time(clock_monotonic, &mts);
  84. break;
  85. default:
  86. goto out;
  87. }
  88. tv->tv_sec = mts.tv_sec;
  89. tv->tv_nsec = mts.tv_nsec;
  90. out:
  91. return retval;
  92. }
  93. #endif
  94. void *cbuf_alloc(unsigned int order)
  95. {
  96. char path[] = "/tmp/cbuf-XXXXXX";
  97. unsigned long size = cbuf_size(order);
  98. void *ret = NULL;
  99. int fd;
  100. fd = mkstemp(path);
  101. if (fd < 0)
  102. return NULL;
  103. if (unlink(path))
  104. goto close;
  105. if (ftruncate(fd, cbuf_size(order)))
  106. goto close;
  107. #ifndef MAP_ANONYMOUS
  108. #define MAP_ANONYMOUS MAP_ANON
  109. #endif
  110. ret = mmap(NULL, size * 2, PROT_NONE, MAP_ANON | MAP_PRIVATE, -1, 0);
  111. if (ret == MAP_FAILED) {
  112. ret = NULL;
  113. goto close;
  114. }
  115. if (mmap(ret, size, PROT_READ | PROT_WRITE, MAP_FIXED | MAP_SHARED,
  116. fd, 0) != ret ||
  117. mmap(ret + size, size, PROT_READ | PROT_WRITE,
  118. MAP_FIXED | MAP_SHARED, fd, 0) != ret + size) {
  119. munmap(ret, size * 2);
  120. ret = NULL;
  121. }
  122. close:
  123. close(fd);
  124. return ret;
  125. }
  126. void cbuf_free(void *ptr, unsigned int order)
  127. {
  128. munmap(ptr, cbuf_size(order) * 2);
  129. }