HP85 GPIB Disk Emulator  1.0
HP85GPIBDiskEmulator
All Data Structures Files Functions Variables Typedefs Enumerations Enumerator Macros Pages
mathio.c
Go to the documentation of this file.
1 
28 #ifdef USER_CODFIG
29 #include "user_config.h"
30 #endif
31 
32 #ifdef PRINTF_TEST
33 #include <stdio.h>
34 #include <stdlib.h>
35 #endif
36 
37 #include <stdint.h>
38 #include <stdarg.h>
39 #include <string.h>
40 
41 #include "mathio.h"
42 
43 #include <math.h>
44 #undef atof
45 
46 // =============================================
47 // MATH/IO
48 // =============================================
49 
50 // =============================================
57 int
58 atodigit(int c,int radix)
59 {
60  int ret = -1;
61  if(c >= '0' && c <= '9')
62  ret = c - '0';
63  else if(c >= 'A' && c <= 'F')
64  ret = c - 'A' + 10;
65  else if(c >= 'a' && c <= 'f')
66  ret = c - 'a' + 10;
67  else return (-1);
68  return((ret >= radix) ? -1 : ret);
69 }
70 
71 
72 // =============================================
80 long atoh(const char *p)
81 {
82  return strtol(p, (char **) NULL, 16);
83 }
84 
85 
86 // =============================================
93 long
94 aton(char *str, int base)
95 {
96  unsigned long num;
97  char *endptr;
98 
99  num = strtol(str, &endptr, base);
100  return(num);
101 }
102 
103 
104 // =============================================
109 MEMSPACE
110 int mul10str(uint8_t *str, int size)
111 {
112 
113  uint16_t d;
114  uint16_t carry = 0;
115  while(size--)
116  {
117  d = *str;
118  d <<= 1;
119  d <<= 1;
120  d += *str;
121  d <<= 1;
122  d += carry;
123  *str = d & 0xff;
124  carry = d >> 8;
125  ++str;
126  }
127  return(carry);
128 }
129 
130 
131 // =============================================
137 MEMSPACE
138 long
139 strtol(const char *nptr, char **endptr, int base)
140 {
141  unsigned long num;
142  int sign;
143  int d;
144 
145  while(*nptr == ' ' || *nptr == '\t')
146  ++nptr;
147  sign = 0;
148  if(*nptr == '-' )
149  {
150  sign = 1;
151  ++nptr;
152  }
153  else if(*nptr == '+' )
154  {
155  ++nptr;
156  }
157 // skip leading zeros
158  while(*nptr == '0')
159  ++nptr;
160  num = 0;
161  while(*nptr)
162  {
163  d = atodigit(*nptr,base);
164  if(d < 0)
165  break;
166  num = num*base;
167  num += d;
168  ++nptr;
169  }
170 
171  if(sign)
172  num = -num;
173  if(endptr)
174  *endptr = (char *) nptr;
175  return(num);
176 }
177 
178 
179 // =============================================
185 MEMSPACE
186 long long
187 strtoll(const char *nptr, char **endptr, int base)
188 {
189  unsigned long long num;
190  int sign;
191  int d;
192 
193  while(*nptr == ' ' || *nptr == '\t')
194  ++nptr;
195  sign = 0;
196  if(*nptr == '-' )
197  {
198  sign = 1;
199  ++nptr;
200  }
201  else if(*nptr == '+' )
202  {
203  ++nptr;
204  }
205 // skip leading zeros
206  while(*nptr == '0')
207  ++nptr;
208 
209  num = 0;
210  while(*nptr)
211  {
212  d = atodigit(*nptr,base);
213  if(d < 0)
214  break;
215  num = num*base;
216  num += d;
217  ++nptr;
218  }
219  if(sign)
220  num = -num;
221  if(endptr)
222  *endptr = (char *) nptr;
223  return(num);
224 }
225 
226 
227 #ifdef __SIZEOF_INT128__
228 // =============================================
234 MEMSPACE
235 __uint128_t
236 strto128(const char *nptr, char **endptr, int base)
237 {
238  __uint128_t num;
239  int sign;
240  int d;
241 
242  while(*nptr == ' ' || *nptr == '\t')
243  ++nptr;
244  sign = 0;
245  if(*nptr == '-' )
246  {
247  sign = 1;
248  ++nptr;
249  }
250  else if(*nptr == '+' )
251  {
252  ++nptr;
253  }
254 
255 // skip leading zeros
256  while(*nptr == '0')
257  ++nptr;
258 
259  num = 0;
260  while(*nptr)
261  {
262  d = atodigit(*nptr,base);
263  if(d < 0)
264  break;
265  num = num*base;
266  num += d;
267  ++nptr;
268  }
269  if(sign)
270  num = -num;
271  if(endptr)
272  *endptr = (char *) nptr;
273  return(num);
274 }
275 #endif
276 
277 // =============================================
282 MEMSPACE
283 int
284 atoi(const char *str)
285 {
286  unsigned long num;
287  num = strtol(str, NULL, 10);
288  return((int)num);
289 }
290 
291 
292 // =============================================
297 MEMSPACE
298 long
299 atol(const char *str)
300 {
301  unsigned long num;
302  num = strtol(str, NULL, 10);
303  return(num);
304 }
305 
306 
307 // =============================================
308 // Floating point I/O helper functions
309 // =============================================
310 #ifdef FLOATIO
311 MEMSPACE
317 double
318 iexp(double num, int exp)
319 {
320  double a;
321  if(exp==0)
322  return(1.0);
323  if(exp <0)
324  {
325  a = 1.0 / num;
326  exp = 1 - exp;
327  }
328  else
329  {
330  exp = exp - 1;
331  a = num;
332  }
333  while(exp)
334  {
335  if(exp & 0x01)
336  num *= a;
337  if(exp >>= 1)
338  a *= a;
339  }
340  return(num);
341 }
342 
343 
344 // =============================================
349 MEMSPACE
350 double
351 scale10(double num, int *exp)
352 {
353  int exp10,exp2;
354  int sign;
355 
356  double scale;
357 
358  if(!num)
359  {
360  *exp = 0;
361  return(0.0);
362  }
363 
364  sign = 0;
365  if(num < 0)
366  {
367  num = -num;
368  sign = 1;
369  }
370 
371 // extract exponent
372  frexp(num, &exp2);
373 // aproximate exponent in base 10
374  exp10 = ((double) exp2) / (double) 3.321928095;
375 
376 // convert scale to 10.0**exp10
377  scale = iexp((double)10.0, exp10);
378 
379 // remove scale
380  num /= scale;
381 
382 // correct for over/under
383  while(num >= (double)10.0)
384  {
385  num /= (double) 10.0;
386  ++exp10;
387  }
388  while(num < (double) 1.0)
389  {
390  num *= (double) 10.0;
391  --exp10;
392  }
393 
394  *exp = exp10;
395  if(sign)
396  return(-num);
397  return(num);
398 }
399 
400 
401 // =============================================
406 MEMSPACE
407 double
408 strtod(const char *nptr, char **endptr)
409 {
410  double num;
411  double frac;
412  double tmp;
413 
414  int digit, power,sign;
415 
416  while(*nptr == ' ' || *nptr == '\t' || *nptr == '\n')
417  ++nptr;
418  sign = 1;
419  if(*nptr == '-')
420  {
421  ++nptr;
422  sign = -1;
423  }
424  else if(*nptr == '+')
425  {
426  ++nptr;
427  }
428 // skip leading zeros
429  while(*nptr == '0')
430  ++nptr;
431  num = 0.0;
432  while(*nptr && isdigit(*nptr))
433  {
434  num *= 10.0; // make room for new digit
435  digit = (*nptr - '0');
436  num += (double) digit;
437  nptr++;
438  }
439 
440  if(*nptr == '.')
441  {
442  ++nptr;
443  frac = 0.1;
444  while(*nptr && isdigit(*nptr))
445  {
446  digit = (*nptr - '0');
447  tmp = (double) digit;
448  num += tmp * frac;
449  frac *= 0.1;
450  nptr++;
451  }
452  }
453  if(sign == -1)
454  num = -num;
455 
456  if(*nptr == 'E' || *nptr == 'e')
457  {
458  nptr++;
459  sign = (*nptr == '-') ? -1 : 1;
460  if(sign == -1 || *nptr == '+')
461  nptr++;
462  power=0;
463  while(isdigit(*nptr))
464  {
465  power *= 10.0;
466  digit = (*nptr - '0');
467  power += (double) digit;
468  nptr++;
469  }
470  if(num)
471  {
472  if(sign<0)
473  power = -power;
474 // iexp - number to integer power
475  num *= iexp(10.0, power);
476  }
477  }
478  if(endptr)
479  *endptr = (char *) nptr;
480  return(num);
481 }
482 
483 
484 // =============================================
488 MEMSPACE
489 double
490 atof(const char *str)
491 {
492  double num;
493  num = strtod(str, NULL);
494  return(num);
495 }
496 #endif // ifdef FLOATIO
atol
MEMSPACE long atol(const char *str)
Convert ASCII string to number in base 10.
Definition: mathio.c:299
strtod
MEMSPACE double strtod(const char *nptr, char **endptr)
MEMSPACE
#define MEMSPACE
Definition: user_config.h:17
strtoll
MEMSPACE long long strtoll(const char *nptr, char **endptr, int base)
Convert ASCII string to number in a given base.
Definition: mathio.c:187
strtol
MEMSPACE long strtol(const char *nptr, char **endptr, int base)
Convert ASCII string to number in a given base.
Definition: mathio.c:139
iexp
MEMSPACE double iexp(double num, int exp)
atof
MEMSPACE double atof(const char *str)
mul10str
MEMSPACE int mul10str(uint8_t *str, int size)
Fast multiply number of any size by 10.
Definition: mathio.c:110
aton
MEMSPACE long aton(char *str, int base)
Convert ASCII string to number in a given base.
Definition: mathio.c:94
atodigit
MEMSPACE int atodigit(int c, int radix)
Convert ASCII character to radix based digit , or -1.
Definition: mathio.c:58
NULL
#define NULL
Definition: user_config.h:85
atoh
MEMSPACE long atoh(const char *p)
Convert ASCII hex string to long integer.
Definition: mathio.c:80
atoi
MEMSPACE int atoi(const char *str)
Convert ASCII string to number in base 10.
Definition: mathio.c:284
mathio.h
Math IO functions, and verious conversion code with floating point support.
isdigit
MEMSPACE int WEAK_ATR isdigit(int c)
test if a character is a digit
Definition: stringsup.c:40
scale10
MEMSPACE double scale10(double num, int *exp)