CPPMyth
Library to interoperate with MythTV server
builtin.c
1 /*
2  * Copyright (C) 2014 Jean-Luc Barriere
3  *
4  * This Program is free software; you can redistribute it and/or modify
5  * it under the terms of the GNU General Public License as published by
6  * the Free Software Foundation; either version 2, or (at your option)
7  * any later version.
8  *
9  * This Program is distributed in the hope that it will be useful,
10  * but WITHOUT ANY WARRANTY; without even the implied warranty of
11  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
12  * GNU General Public License for more details.
13  *
14  * You should have received a copy of the GNU General Public License
15  * along with this program; see the file COPYING. If not, write to
16  * the Free Software Foundation, 51 Franklin Street, Fifth Floor, Boston,
17  * MA 02110-1301 USA
18  * http://www.gnu.org/copyleft/gpl.html
19  *
20  */
21 
22 #include "builtin.h"
23 
24 #include <ctype.h>
25 #include <stdlib.h>
26 #include <stdio.h>
27 #include <errno.h>
28 
29 int string_to_int64(const char *str, int64_t *num)
30 {
31  uint64_t val = 0;
32  int sign = 1;
33  uint64_t limit = INT64_MAX;
34 
35  if (str == NULL)
36  return -(EINVAL);
37  while (isspace(*str))
38  {
39  ++str;
40  }
41 
42  if (*str && (*str == '-'))
43  {
44  ++str;
45  sign = -1;
46  }
47  while (*str && !isspace(*str))
48  {
49  if (!isdigit(*str))
50  return -(EINVAL);
51  val *= 10;
52  val += ((*str) - '0');
53  /*
54  * Check and make sure we are still under the limit (this is
55  * an absolute value limit, sign will be applied later).
56  */
57  if (val > limit)
58  return -(ERANGE);
59  str++;
60  }
61 
62  *num = (int64_t) (sign * val);
63  return 0;
64 }
65 
66 int string_to_int32(const char *str, int32_t *num)
67 {
68  uint64_t val = 0;
69  int sign = 1;
70  uint64_t limit = INT32_MAX;
71 
72  if (str == NULL)
73  return -(EINVAL);
74  while (isspace(*str))
75  {
76  ++str;
77  }
78 
79  if (*str && (*str == '-'))
80  {
81  ++str;
82  sign = -1;
83  }
84  while (*str && !isspace(*str))
85  {
86  if (!isdigit(*str))
87  return -(EINVAL);
88  val *= 10;
89  val += ((*str) - '0');
90  /*
91  * Check and make sure we are still under the limit (this is
92  * an absolute value limit, sign will be applied later).
93  */
94  if (val > limit)
95  return -(ERANGE);
96  str++;
97  }
98 
99  *num = (int32_t) (sign * val);
100  return 0;
101 }
102 
103 int string_to_int16(const char *str, int16_t *num)
104 {
105  int32_t val;
106  int err;
107 
108  if ((err = string_to_int32(str, &val)))
109  return err;
110  if (val > 32767 || val < -32768)
111  return -(ERANGE);
112 
113  *num = (int16_t) val;
114  return 0;
115 }
116 
117 int string_to_int8(const char *str, int8_t *num)
118 {
119  int32_t val;
120  int err;
121 
122  if ((err = string_to_int32(str, &val)))
123  return err;
124  if (val > 127 || val < -128)
125  return -(ERANGE);
126 
127  *num = (int8_t) val;
128  return 0;
129 }
130 
131 int string_to_uint32(const char *str, uint32_t *num)
132 {
133  uint64_t val = 0;
134  uint64_t limit = UINT32_MAX;
135 
136  if (str == NULL)
137  return -(EINVAL);
138  while (isspace(*str))
139  {
140  ++str;
141  }
142 
143  while (*str && !isspace(*str))
144  {
145  if (!isdigit(*str))
146  return -(EINVAL);
147  val *= 10;
148  val += ((*str) - '0');
149  /*
150  * Check and make sure we are still under the limit (this is
151  * an absolute value limit, sign will be applied later).
152  */
153  if (val > limit)
154  return -(ERANGE);
155  str++;
156  }
157 
158  *num = (uint32_t) val;
159  return 0;
160 }
161 
162 int string_to_uint16(const char *str, uint16_t *num)
163 {
164  uint32_t val;
165  int err;
166 
167  if ((err = string_to_uint32(str, &val)))
168  return err;
169  if (val > 65535)
170  return -(ERANGE);
171 
172  *num = (uint16_t) val;
173  return 0;
174 }
175 
176 int string_to_uint8(const char *str, uint8_t *num)
177 {
178  uint32_t val;
179  int err;
180 
181  if ((err = string_to_uint32(str, &val)))
182  return err;
183  if (val > 255)
184  return -(ERANGE);
185 
186  *num = (uint8_t) val;
187  return 0;
188 }
189 
190 time_t __timegm(struct tm *utctime_tm)
191 {
192  time_t time;
193  struct tm adj_tm, chk_tm;
194 
195  adj_tm.tm_year = utctime_tm->tm_year;
196  adj_tm.tm_mon = utctime_tm->tm_mon;
197  adj_tm.tm_mday = utctime_tm->tm_mday;
198  adj_tm.tm_hour = utctime_tm->tm_hour;
199  adj_tm.tm_min = utctime_tm->tm_min;
200  adj_tm.tm_sec = utctime_tm->tm_sec;
201  adj_tm.tm_isdst = -1;
202 
203  for (;;)
204  {
205  time = mktime(&adj_tm);
206  if (time == INVALID_TIME)
207  return time;
208  if (NULL == gmtime_r(&time, &chk_tm))
209  return INVALID_TIME;
210  if (chk_tm.tm_min == utctime_tm->tm_min &&
211  chk_tm.tm_hour == utctime_tm->tm_hour &&
212  chk_tm.tm_mday == utctime_tm->tm_mday &&
213  chk_tm.tm_mon == utctime_tm->tm_mon &&
214  chk_tm.tm_year == utctime_tm->tm_year)
215  break;
216  adj_tm.tm_min += utctime_tm->tm_min - chk_tm.tm_min;
217  adj_tm.tm_hour += utctime_tm->tm_hour - chk_tm.tm_hour;
218  adj_tm.tm_mday += utctime_tm->tm_mday - chk_tm.tm_mday;
219  adj_tm.tm_mon += utctime_tm->tm_mon - chk_tm.tm_mon;
220  adj_tm.tm_year += utctime_tm->tm_year - chk_tm.tm_year;
221  }
222  return time;
223 }
224 
225 int string_to_time(const char *str, time_t *time)
226 {
227  struct tm time_tm;
228  int i, len, format, isutc;
229  char *yyyy, *MM, *dd, *hh, *mm, *ss;
230  char buf[TIMESTAMP_UTC_LEN + 1];
231 
232  if (*str == '\0')
233  {
234  /* empty string */
235  *time = INVALID_TIME;
236  return 0;
237  }
238  memset(buf, 0, sizeof(buf));
239  strncpy(buf, str, sizeof(buf) - 1);
240  len = strlen(buf);
241 
242  switch (len)
243  {
244  case TIMESTAMP_UTC_LEN:
245  if (((buf[4] != '-') || (buf[7] != '-') || (buf[10] != 'T') ||
246  (buf[13] != ':') || (buf[16] != ':') || (buf[19] != 'Z')))
247  {
248  fprintf(stderr, "%s: string is badly formed '%s'\n", __FUNCTION__, buf);
249  goto err;
250  }
251  format = 1;
252  isutc = 1;
253  break;
254  case TIMESTAMP_LEN:
255  if (((buf[4] != '-') || (buf[7] != '-') || (buf[10] != 'T') ||
256  (buf[13] != ':') || (buf[16] != ':')))
257  {
258  fprintf(stderr, "%s: string is badly formed '%s'\n", __FUNCTION__, buf);
259  goto err;
260  }
261  format = 2;
262  isutc = 0;
263  break;
264  case DATESTAMP_LEN:
265  if (((buf[4] != '-') || (buf[7] != '-')))
266  {
267  fprintf(stderr, "%s: string is badly formed '%s'\n", __FUNCTION__, buf);
268  goto err;
269  }
270  format = 3;
271  isutc = 0;
272  break;
273  default:
274  fprintf(stderr, "%s: string is not a timestamp '%s'\n", __FUNCTION__, buf);
275  goto err;
276  break;
277  }
278 
279  yyyy = buf;
280  MM = buf + 5;
281  dd = buf + 8;
282  hh = buf + 11;
283  mm = buf + 14;
284  ss = buf + 17;
285 
286  yyyy[4] = MM[2] = dd[2] = hh[2] = mm[2] = ss[2] = '\0';
287 
288  for (i = 0; i < len; ++i)
289  {
290  if (buf[i] && !isdigit(buf[i]))
291  {
292  fprintf(stderr, "%s: expected numeral at '%s'[%d]\n", __FUNCTION__, str, i);
293  goto err;
294  }
295  }
296 
297  time_tm.tm_isdst = -1;
298  time_tm.tm_year = atoi(yyyy) - 1900;
299  time_tm.tm_mon = atoi(MM) - 1;
300  if (time_tm.tm_mon > 11)
301  {
302  fprintf(stderr, "%s: month value too big '%s'\n", __FUNCTION__, str);
303  goto err;
304  }
305  time_tm.tm_mday = atoi(dd);
306  if (time_tm.tm_mday > 31)
307  {
308  fprintf(stderr, "%s: day value too big '%s'\n", __FUNCTION__, str);
309  goto err;
310 
311  }
312 
313  if (format == 3)
314  {
315  time_tm.tm_hour = time_tm.tm_min = time_tm.tm_sec = 0;
316  *time = mktime(&time_tm);
317  return 0;
318  }
319 
320  time_tm.tm_hour = atoi(hh);
321  if (time_tm.tm_hour > 23)
322  {
323  fprintf(stderr, "%s: hour value too big '%s'\n", __FUNCTION__, str);
324  goto err;
325  }
326  time_tm.tm_min = atoi(mm);
327  if (time_tm.tm_min > 59)
328  {
329  fprintf(stderr, "%s: minute value too big '%s'\n", __FUNCTION__, str);
330  goto err;
331  }
332  time_tm.tm_sec = atoi(ss);
333  if (time_tm.tm_sec > 59)
334  {
335  fprintf(stderr, "%s: second value too big '%s'\n", __FUNCTION__, str);
336  goto err;
337  }
338 
339  if (isutc)
340  *time = timegm(&time_tm);
341  else
342  *time = mktime(&time_tm);
343  return 0;
344 
345 err:
346  *time = INVALID_TIME;
347  return -(EINVAL);
348 }
349 
350 void time_to_iso8601utc(time_t time, char *str)
351 {
352  struct tm time_tm;
353 
354  if (time == INVALID_TIME || NULL == gmtime_r(&time, &time_tm))
355  {
356  str[0] = '\0';
357  return;
358  }
359  sprintf(str, "%4.4d-%2.2d-%2.2dT%2.2d:%2.2d:%2.2dZ",
360  time_tm.tm_year + 1900,
361  time_tm.tm_mon + 1,
362  time_tm.tm_mday,
363  time_tm.tm_hour,
364  time_tm.tm_min,
365  time_tm.tm_sec);
366 }
367 
368 void time_to_iso8601(time_t time, char *str)
369 {
370  struct tm time_tm;
371 
372  if (time == INVALID_TIME || NULL == localtime_r(&time, &time_tm))
373  {
374  str[0] = '\0';
375  return;
376  }
377  sprintf(str, "%4.4d-%2.2d-%2.2dT%2.2d:%2.2d:%2.2d",
378  time_tm.tm_year + 1900,
379  time_tm.tm_mon + 1,
380  time_tm.tm_mday,
381  time_tm.tm_hour,
382  time_tm.tm_min,
383  time_tm.tm_sec);
384 }
385 
386 void time_to_isodate(time_t time, char *str)
387 {
388  struct tm time_tm;
389 
390  if (time == INVALID_TIME || NULL == localtime_r(&time, &time_tm))
391  {
392  str[0] = '\0';
393  return;
394  }
395  sprintf(str, "%4.4d-%2.2d-%2.2d",
396  time_tm.tm_year + 1900,
397  time_tm.tm_mon + 1,
398  time_tm.tm_mday);
399 }