lsquic_util.c revision 10c492f0
1/* Copyright (c) 2017 - 2018 LiteSpeed Technologies Inc. See LICENSE. */ 2/* 3 * Utility functions 4 */ 5 6#include <ctype.h> 7#include <stdint.h> 8#include <stdio.h> 9#include <string.h> 10#include <time.h> 11#ifndef WIN32 12#include <sys/time.h> 13#include <unistd.h> 14#else 15#include <vc_compat.h> 16#endif 17 18#if !(defined(_POSIX_TIMERS) && _POSIX_TIMERS > 0) && defined(__APPLE__) 19#include <mach/mach_time.h> 20#endif 21 22#include "lsquic_int_types.h" 23#include "lsquic_util.h" 24 25 26#if defined(__APPLE__) 27static mach_timebase_info_data_t timebase; 28#endif 29#if defined(WIN32) 30static LARGE_INTEGER perf_frequency; 31#endif 32 33void 34lsquic_init_timers (void) 35{ 36#if defined(__APPLE__) 37 mach_timebase_info(&timebase); 38#endif 39#if defined(WIN32) 40 QueryPerformanceFrequency(&perf_frequency); 41#endif 42} 43 44 45lsquic_time_t 46lsquic_time_now (void) 47{ 48#if defined(_POSIX_TIMERS) && _POSIX_TIMERS > 0 49 struct timespec ts; 50 (void) clock_gettime(CLOCK_MONOTONIC, &ts); 51 return (lsquic_time_t) ts.tv_sec * 1000000 + ts.tv_nsec / 1000; 52#elif defined(__APPLE__) 53 lsquic_time_t t = mach_absolute_time(); 54 t *= timebase.numer; 55 t /= timebase.denom; 56 t /= 1000; 57 return t; 58#elif defined(WIN32) 59 LARGE_INTEGER counter; 60 lsquic_time_t t; 61 QueryPerformanceCounter(&counter); 62 t = counter.QuadPart; 63 t *= 1000000; 64 t /= perf_frequency.QuadPart; 65 return t; 66#else 67# warn Monotonically increasing clock is not available on this platform 68 struct timeval tv; 69 (void) gettimeofday(&tv, NULL); 70 return (lsquic_time_t) tv.tv_sec * 1000000 + tv.tv_usec; 71#endif 72} 73 74 75int 76lsquic_is_zero (const void *pbuf, size_t bufsz) 77{ 78 const unsigned char *buf, *end; 79 const unsigned long *buf_ul; 80 unsigned n_ul; 81 unsigned long n_non_zero; 82 83 buf = pbuf; 84 end = buf + bufsz; 85 buf_ul = (unsigned long *) buf; 86 n_ul = bufsz / sizeof(buf_ul[0]); 87 buf += n_ul * sizeof(buf_ul[0]); 88 n_non_zero = 0; 89 90 while (n_ul--) 91 n_non_zero |= buf_ul[n_ul]; 92 93 while (buf < end) 94 n_non_zero |= *buf++; 95 96 return n_non_zero == 0; 97} 98 99 100/* XXX this function uses static buffer. Replace it with hexdump() if possible */ 101char *get_bin_str(const void *s, size_t len, size_t max_display_len) 102{ 103 const unsigned char *p, *pEnd; 104 char *pOutput; 105 size_t lenOrg = len; 106 static char str[512 * 2 + 40] = {0}; 107 108 /** 109 * We alloc fixed size buffer, at most max_display_len is 512 110 */ 111 size_t fit_display_len = (max_display_len > 512 ? 512 : max_display_len); 112 if (len > fit_display_len) 113 len = fit_display_len; 114 115 pOutput = &str[0] + sprintf(str, "(%zd/%zd)=0x", len, lenOrg); 116 117 for(p = s, pEnd = (unsigned char*)s + len; p < pEnd; ++p) 118 { 119 sprintf(pOutput, "%02X", *p); 120 pOutput += 2; 121 } 122 if (lenOrg > len) 123 { 124 sprintf(pOutput, "..."); 125 pOutput += 3; 126 } 127 return str; 128} 129 130 131size_t 132hexdump (const void *src_void, size_t src_sz, char *out, size_t out_sz) 133{ 134/* Ruler: 135 * 136 6 31 57 73 137 | | | | 1380000 00 01 02 03 04 05 06 07 08 09 0A 0B 0C 0D 0E 0F |................| 139 * 140 */ 141#define LINE_SIZE (74 + 1 /* newline */) 142 const unsigned char *src = src_void; 143 const unsigned char *const src_end = src + src_sz; 144 char *const out_end = out + out_sz; 145 unsigned line = 0; 146 147 while (src < src_end && out_end - out >= LINE_SIZE) 148 { 149 const unsigned char *limit = src + 16; 150 if (limit > src_end) 151 limit = src_end; 152 unsigned hex_off = 6; 153 unsigned alpha_off = 57; 154 sprintf(out, "%04x", line++); 155 out[4] = ' '; 156 out[5] = ' '; 157 while (src < limit) 158 { 159 sprintf(out + hex_off, "%02X ", *src); 160 sprintf(out + alpha_off, "%c", isprint(*src) ? *src : '.'); 161 hex_off += 3; 162 out[hex_off] = ' '; 163 hex_off += 30 == hex_off; 164 out[hex_off] = ' '; 165 ++alpha_off; 166 out[alpha_off] = ' '; 167 ++src; 168 } 169 memset(out + hex_off, ' ', 56 - hex_off); 170 memset(out + alpha_off, '.', 73 - alpha_off); 171 out[56] = '|'; 172 out[73] = '|'; 173 out[74] = '\n'; 174 out += LINE_SIZE; 175 } 176 177 if (out < out_end) 178 *out = '\0'; 179 else 180 out_end[-1] = '\0'; 181 182 return out + out_sz - out_end; 183} 184