fuse_kafka
src/util.c
Go to the documentation of this file.
00001 
00004 #ifndef UTIL_C
00005 #define UTIL_C
00006 #include <stdlib.h>
00007 #include <string.h>
00008 #include <unistd.h>
00009 #include <sys/stat.h>
00010 #include <sys/types.h>
00011 #include <sys/time.h>
00012 #include <time.h>
00013 #include <math.h>
00014 #include <stdio.h>
00015 int strcmp(const char* s1, const char* s2)
00016 {
00017     while(*s1 && (*s1==*s2))
00018         s1++,s2++;
00019     return *(const unsigned char*)s1-*(const unsigned char*)s2;
00020 }
00021 int touch(char* path, char* str)
00022 {
00023     FILE* f = fopen(path, "w");
00024     if(f == NULL) return 0;
00025     fwrite(str, strlen(str), 1, f);
00026     fclose(f);
00027     return 1;
00028 }
00029 #define BASE64_CHARS \
00030     "ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789+/"
00031 
00038 char* base64(const unsigned char* str, int n)
00039 {
00040     static const char* base64chars = BASE64_CHARS;
00041     if(str == NULL) return NULL;
00042     int i = 0, j, k;
00043     int added = n % 3;
00044     added = (added == 0? 0: (added == 1? 2 : 1));
00045     int size = (n + added) * 8 / 6;
00046     char* result = (char*) calloc(sizeof(char), size + added + 1);
00047     result[size] = 0;
00048     for(j = 0; j < (n + added); j++)
00049     {
00050         char c = j < size ? str[j] : 0;
00051         for(k = 0; k < 8; k++)
00052         {
00053             int i_k = i + k;
00054             int i_k_6 = i_k / 6;
00055             if((i_k % 6) == 0)
00056             {
00057                 if(i_k_6 >= 1) result[i_k_6 - 1] = base64chars[result[i_k_6 - 1]];
00058                 result[i_k_6] = 0;
00059             }
00060             result[i_k_6] = (result[i_k_6] << 1) | ((c >> (7 - k)) & 1);
00061         }
00062         i += 8;
00063     }
00064     /* for last char, base64chars might not have been called: */
00065     if(added == 0 && size > (added + 1)) result[size - 1 - added] = base64chars[result[size - 1 - added]];
00066     for(j = 0; j < added; j++) result[size - 1 - j] = '=';
00067     return result;
00068 }
00069 int* get_command_line_size()
00070 {
00071     static int command_line_size = 256;
00072     return &command_line_size;
00073 }
00080 static char* get_command_line(int pid)
00081 {
00082     size_t size = *get_command_line_size();;
00083     size_t i = 0;
00084     char* path;
00085     FILE* f;
00086     char c;
00087     char* command_line = (char*) malloc(size * sizeof(char));
00088     char* b64;
00089     asprintf(&path, "/proc/%d/cmdline", pid);
00090     if((f = fopen(path, "r")) != NULL)
00091     {
00092         while((c = fgetc(f)) != EOF)
00093         {
00094             if(c == 0) c = ' ';
00095             if(i >= (size - 1))
00096             {
00097                 size += *get_command_line_size();
00098                 command_line = (char*) realloc(
00099                         command_line, size * sizeof(char));
00100             }
00101             command_line[i++] = c;
00102         }
00103         fclose(f);
00104     }
00105     command_line[i] = 0;
00106     free(path);
00107     b64 = base64(command_line, strlen(command_line));
00108     free(command_line);
00109     return b64;
00110 }
00111 #ifdef MINGW_VER
00112 void set_timezone(char* timestamp, int minutes)
00113 {
00114     long hrs, mins;
00115     timestamp[23] = minutes < 0 ? '+' : '-';
00116     if(minutes < 0) minutes = -minutes;
00117     hrs = minutes / 60;
00118     mins = minutes % 60;
00119     timestamp[24] += hrs/10;
00120     timestamp[25] += hrs%10;
00121     timestamp[26] += mins/10;
00122     timestamp[27] += mins%10;
00123 }
00124 #endif
00125 void set_msec(char* timestamp, suseconds_t msec)
00126 {
00127     timestamp[22] = '0' + msec % 10;
00128     timestamp[21] = '0' + (msec % 100) / 10;
00129     timestamp[20] = '0' + (msec % 1000) / 100;
00130 }
00136 void set_timestamp(char* timestamp)
00137 {
00138     struct tm tm;
00139     struct tm* tmp;
00140     struct timeval tv;
00141     struct timezone tz;
00142     gettimeofday(&tv, &tz);
00143 #ifdef MINGW_VER
00144 #define TZ "+0000"
00145     time_t ltime;
00146     time(&ltime);
00147     tmp = localtime(&ltime);
00148 #else
00149 #define TZ "%z"
00150     localtime_r(&tv.tv_sec, &tm);
00151     tmp = &tm;
00152 #endif
00153     size_t size = strftime(timestamp, strlen(timestamp), "%Y-%m-%dT%H:%M:%S.000" TZ, tmp);
00154     set_msec(timestamp, tv.tv_usec / 1000);
00155 #ifdef MINGW_VER
00156     set_timezone(timestamp, tz.tz_minuteswest);
00157 #endif
00158 }
00159 #ifdef MINGW_VER
00160 /* TODO implement */
00161 int fnmatch(char* pattern, char* string, int flags)
00162 {
00163     return 0;
00164 }
00165 int flock(int fd, int operation)
00166 {
00167     return 0;
00168 }
00169 int fchdir(int fd)
00170 {
00171     return 0;
00172 }
00173 #include <errno.h>
00174 /* implementation from 
00175 http://stackoverflow.com/questions/27369580/codeblocks-and-c-undefined-reference-to-getline
00176 */
00177 ssize_t getdelim(char **linep, size_t *n, int delim, FILE *fp){
00178     int ch;
00179     size_t i = 0;
00180     if(!linep || !n || !fp){
00181         errno = EINVAL;
00182         return -1;
00183     }
00184     if(*linep == NULL){
00185         if(NULL==(*linep = malloc(*n=128))){
00186             *n = 0;
00187             errno = ENOMEM;
00188             return -1;
00189         }
00190     }
00191     while((ch = fgetc(fp)) != EOF){
00192         if(i + 1 >= *n){
00193             char *temp = realloc(*linep, *n + 128);
00194             if(!temp){
00195                 errno = ENOMEM;
00196                 return -1;
00197             }
00198             *n += 128;
00199             *linep = temp;
00200         }
00201         (*linep)[i++] = ch;
00202         if(ch == delim)
00203             break;
00204     }
00205     (*linep)[i] = '\0';
00206     return !i && ch == EOF ? -1 : i;
00207 }
00208 ssize_t getline(char **linep, size_t *n, FILE *fp){
00209     return getdelim(linep, n, '\n', fp);
00210 }
00211 #endif
00212 
00218 int get_limit(int argc, char** argv)
00219 {
00220     int i = 0;
00221     for(; i < argc; i++) if(!strcmp(argv[i], "--")) break;
00222     return i;
00223 }
00227 char* array_to_container_string(char** array, size_t n, char open_char,
00228         char close_char, char sep1, char sep2)
00229 {
00230     int i = 0;
00231     char* str = (char*) malloc(3);
00232     int k = sprintf(str, "%c", open_char);
00233     if(array != NULL)
00234         for(i = 0; i < n; i++)
00235         {
00236             str = realloc(str, k + 1 + strlen(array[i]) + 2 + 2);
00237             k += sprintf(str + k, "\"%s\"", array[i]);
00238             if(i != n-1) k += sprintf(str + k, "%c ", i % 2 ? sep2 : sep1);
00239         }
00240     sprintf(str + k, "%c", close_char);
00241     return str;
00242 }
00243 int* falloc_fails()
00244 {
00245     static int result = 0;
00246     return &result;
00247 }
00248 int* fcalloc_fails()
00249 {
00250     static int result = 0;
00251     return &result;
00252 }
00253 void* fmalloc(size_t size)
00254 {
00255     if(*falloc_fails()) return NULL;
00256     return malloc(size);
00257 }
00258 void* fcalloc(size_t nmemb, size_t size)
00259 {
00260     if(*falloc_fails() || *fcalloc_fails()) return NULL;
00261     return calloc(nmemb, size);
00262 }
00263 void* frealloc(void* ptr, size_t size)
00264 {
00265     if(*falloc_fails()) return NULL;
00266     return realloc(ptr, size);
00267 }
00268 char* integer_concat(char* prefix, int i, char* suffix)
00269 {
00270     int size = (strlen(prefix) + ceil(log10(i))+1 + strlen(suffix))*sizeof(char);
00271     char *str = (char*) fmalloc((int)(size) + 1);
00272     if(str != NULL)
00273         snprintf(str, size, "%s%d%s", prefix, i, suffix);
00274     return str;
00275 }
00279 char* concat(char* a, char* b)
00280 {
00281     if(a == NULL || b == NULL) return NULL;
00282     int length = strlen(a) + strlen(b) + 2;
00283     char* result = (char*) malloc(length * sizeof(char));
00284     if(result == NULL) return NULL;
00285     result[length - 1] = 0;
00286     char* middle = result + strlen(a);
00287     strcpy(result, a);
00288     middle[0] = '/';
00289     strcpy(++middle, b);
00290     return result;
00291 }
00292 int* fk_sleep_return_value()
00293 {
00294     static int value = 0;
00295     return &value;
00296 }
00297 int* fk_sleep_enabled()
00298 {
00299     static int enabled = 1;
00300     return &enabled;
00301 }
00302 int fk_sleep(int n)
00303 {
00304     if(*(fk_sleep_enabled())) return sleep(n);
00305     return *(fk_sleep_return_value());
00306 }
00307 #ifdef _WIN32
00308 #define mkdir( D, M ) _mkdir( D )
00309 #endif 
00310 static void mkdir_p(const char *dir) {
00311         char tmp[256];
00312         char *p = NULL;
00313         size_t len;
00314 
00315         snprintf(tmp, sizeof(tmp),"%s",dir);
00316         len = strlen(tmp);
00317         if(tmp[len - 1] == '/')
00318                 tmp[len - 1] = 0;
00319         for(p = tmp + 1; *p; p++)
00320                 if(*p == '/') {
00321                         *p = 0;
00322                         mkdir(tmp, S_IRWXU);
00323                         *p = '/';
00324                 }
00325         mkdir(tmp, S_IRWXU);
00326 }
00327 #define DO_AS_CALLER(action) \
00328     struct fuse_context* __context = fuse_get_context(); \
00329     gid_t __gid = getegid(); \
00330     uid_t __uid = geteuid(); \
00331     setegid(__context->gid); \
00332     seteuid(__context->uid); \
00333     action \
00334     seteuid(__uid); \
00335     setegid(__gid);
00336 #endif
 All Data Structures Files Functions Variables Defines