Coverage Report

Created: 2024-06-18 06:05

/src/leptonica/src/utils1.c
Line
Count
Source (jump to first uncovered line)
1
/*====================================================================*
2
 -  Copyright (C) 2001 Leptonica.  All rights reserved.
3
 -
4
 -  Redistribution and use in source and binary forms, with or without
5
 -  modification, are permitted provided that the following conditions
6
 -  are met:
7
 -  1. Redistributions of source code must retain the above copyright
8
 -     notice, this list of conditions and the following disclaimer.
9
 -  2. Redistributions in binary form must reproduce the above
10
 -     copyright notice, this list of conditions and the following
11
 -     disclaimer in the documentation and/or other materials
12
 -     provided with the distribution.
13
 -
14
 -  THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
15
 -  ``AS IS'' AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
16
 -  LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
17
 -  A PARTICULAR PURPOSE ARE DISCLAIMED.  IN NO EVENT SHALL ANY
18
 -  CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL,
19
 -  EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO,
20
 -  PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR
21
 -  PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY
22
 -  OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING
23
 -  NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS
24
 -  SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
25
 *====================================================================*/
26
27
/*!
28
 * \file utils1.c
29
 * <pre>
30
 *
31
 *       ------------------------------------------
32
 *       This file has these utilities:
33
 *         - error, warning and info messages
34
 *         - redirection of stderr
35
 *         - low-level endian conversions
36
 *         - file corruption operations
37
 *         - random and prime number operations
38
 *         - 64-bit hash functions
39
 *         - leptonica version number accessor
40
 *         - timing and date operations
41
 *       ------------------------------------------
42
 *
43
 *       Control of error, warning and info messages
44
 *           l_int32    setMsgSeverity()
45
 *
46
 *       Error return functions, invoked by macros
47
 *           l_int32    returnErrorInt()
48
 *           l_float32  returnErrorFloat()
49
 *           void      *returnErrorPtr()
50
 *           l_int32    returnErrorInt1()
51
 *           l_float32  returnErrorFloat1()
52
 *           void      *returnErrorPtr1()
53
 *
54
 *       Runtime redirection of stderr
55
 *           void leptSetStderrHandler()
56
 *           void lept_stderr()
57
 *
58
 *       Test files for equivalence
59
 *           l_int32    filesAreIdentical()
60
 *
61
 *       Byte-swapping data conversion
62
 *           l_uint16   convertOnBigEnd16()
63
 *           l_uint32   convertOnBigEnd32()
64
 *           l_uint16   convertOnLittleEnd16()
65
 *           l_uint32   convertOnLittleEnd32()
66
 *
67
 *       File corruption and byte replacement operations
68
 *           l_int32    fileCorruptByDeletion()
69
 *           l_int32    fileCorruptByMutation()
70
 *           l_int32    fileReplaceBytes()
71
 *
72
 *       Generate random integer in given interval
73
 *           l_int32    genRandomIntOnInterval()
74
 *
75
 *       Simple math functions
76
 *           l_int32    lept_roundftoi()
77
 *           l_int32    lept_floor()
78
 *           l_int32    lept_ceiling()
79
 *
80
 *       64-bit hash functions
81
 *           l_int32    l_hashStringToUint64()
82
 *           l_int32    l_hashStringToUint64Fast()
83
 *           l_int32    l_hashPtToUint64()
84
 *           l_int32    l_hashFloat64ToUint64()
85
 *
86
 *       Prime finders
87
 *           l_int32    findNextLargerPrime()
88
 *           l_int32    lept_isPrime()
89
 *
90
 *       Gray code conversion
91
 *           l_uint32   convertIntToGrayCode()
92
 *           l_uint32   convertGrayCodeToInt()
93
 *
94
 *       Leptonica version number
95
 *           char      *getLeptonicaVersion()
96
 *
97
 *       Timing
98
 *           void       startTimer()
99
 *           l_float32  stopTimer()
100
 *           L_TIMER    startTimerNested()
101
 *           l_float32  stopTimerNested()
102
 *           void       l_getCurrentTime()
103
 *           L_WALLTIMER  *startWallTimer()
104
 *           l_float32  stopWallTimer()
105
 *           void       l_getFormattedDate()
106
 *
107
 *  For all issues with cross-platform development, see utils2.c.
108
 * </pre>
109
 */
110
111
#ifdef HAVE_CONFIG_H
112
#include <config_auto.h>
113
#endif  /* HAVE_CONFIG_H */
114
115
#ifdef _WIN32
116
#include <windows.h>
117
#endif  /* _WIN32 */
118
119
#include <time.h>
120
#include "allheaders.h"
121
#include <math.h>
122
123
    /* Global for controlling message output at runtime */
124
LEPT_DLL l_int32  LeptMsgSeverity = DEFAULT_SEVERITY;
125
126
#define  DEBUG_SEV     0
127
128
/*----------------------------------------------------------------------*
129
 *                Control of error, warning and info messages           *
130
 *----------------------------------------------------------------------*/
131
/*!
132
 * \brief   setMsgSeverity()
133
 *
134
 * \param[in]    newsev
135
 * \return  oldsev
136
 *
137
 * <pre>
138
 * Notes:
139
 *      (1) setMsgSeverity() allows the user to specify the desired
140
 *          message severity threshold.  Messages of equal or greater
141
 *          severity will be output.  The previous message severity is
142
 *          returned when the new severity is set.
143
 *      (2) If L_SEVERITY_EXTERNAL is passed, then the severity will be
144
 *          obtained from the LEPT_MSG_SEVERITY environment variable.
145
 * </pre>
146
 */
147
l_int32
148
setMsgSeverity(l_int32  newsev)
149
0
{
150
0
l_int32  oldsev;
151
0
char    *envsev;
152
153
0
    oldsev = LeptMsgSeverity;
154
0
    if (newsev == L_SEVERITY_EXTERNAL) {
155
0
        envsev = getenv("LEPT_MSG_SEVERITY");
156
0
        if (envsev) {
157
0
            LeptMsgSeverity = atoi(envsev);
158
#if DEBUG_SEV
159
            L_INFO("message severity set to external\n", "setMsgSeverity");
160
#endif  /* DEBUG_SEV */
161
0
        } else {
162
#if DEBUG_SEV
163
            L_WARNING("environment var LEPT_MSG_SEVERITY not defined\n",
164
                      "setMsgSeverity");
165
#endif  /* DEBUG_SEV */
166
0
        }
167
0
    } else {
168
0
        LeptMsgSeverity = newsev;
169
#if DEBUG_SEV
170
        L_INFO("message severity set to %d\n", "setMsgSeverity", newsev);
171
#endif  /* DEBUG_SEV */
172
0
    }
173
174
0
    return oldsev;
175
0
}
176
177
178
/*----------------------------------------------------------------------*
179
 *                Error return functions, invoked by macros             *
180
 *----------------------------------------------------------------------*
181
 *                                                                      *
182
 *    (1) These error functions print messages to stderr and allow      *
183
 *        exit from the function that called them.                      *
184
 *    (2) They must be invoked only by the three argument macros        *
185
 *           ERROR_INT, ERROR_FLOAT, ERROR_PTR                          *
186
 *        or the four argument macros                                   *
187
 *           ERROR_INT_1, ERROR_FLOAT_1, ERROR_PTR_1                    *
188
 *        which are in environ.h.                                       *
189
 *    (3) The print output can be disabled at compile time, either      *
190
 *        by using -DNO_CONSOLE_IO or by setting LeptMsgSeverity.       *
191
 *----------------------------------------------------------------------*/
192
/*!
193
 * \brief   returnErrorInt()
194
 *
195
 * \param[in]    msg        error message
196
 * \param[in]    procname   use __func__
197
 * \param[in]    ival       return error val (typically 1 for an error)  
198
 * \return  ival
199
 */
200
l_int32
201
returnErrorInt(const char  *msg,
202
               const char  *procname,
203
               l_int32      ival)
204
1.29k
{
205
1.29k
    lept_stderr("Error in %s: %s\n", procname, msg);
206
1.29k
    return ival;
207
1.29k
}
208
209
210
/*!
211
 * \brief   returnErrorFloat()
212
 *
213
 * \param[in]    msg        error message
214
 * \param[in]    procname   use __func__
215
 * \param[in]    fval       return float error val
216
 * \return  fval
217
 */
218
l_float32
219
returnErrorFloat(const char  *msg,
220
                 const char  *procname,
221
                 l_float32    fval)
222
0
{
223
0
    lept_stderr("Error in %s: %s\n", procname, msg);
224
0
    return fval;
225
0
}
226
227
228
/*!
229
 * \brief   returnErrorPtr()
230
 *
231
 * \param[in]    msg        error message
232
 * \param[in]    procname   use __func__
233
 * \param[in]    pval       return error val (typically null for an error)
234
 * \return  pval
235
 */
236
void *
237
returnErrorPtr(const char  *msg,
238
               const char  *procname,
239
               void        *pval)
240
10.3k
{
241
10.3k
    lept_stderr("Error in %s: %s\n", procname, msg);
242
10.3k
    return pval;
243
10.3k
}
244
245
246
/*!
247
 * \brief   returnErrorInt1()
248
 *
249
 * \param[in]    msg        error message
250
 * \param[in]    arg        additional error message argument
251
 *                          (will be appended to the error message)
252
 * \param[in]    procname   use __func__
253
 * \param[in]    ival       return error val; typically 1 for an error return
254
 * \return  ival   typically 1 for an error return
255
 */
256
l_int32
257
returnErrorInt1(const char  *msg,
258
                const char  *arg,
259
                const char  *procname,
260
                l_int32      ival)
261
0
{
262
0
    lept_stderr("Leptonica Error in %s: %s: %s\n", procname, msg, arg);
263
0
    return ival;
264
0
}
265
266
267
/*!
268
 * \brief   returnErrorFloat1()
269
 *
270
 * \param[in]    msg        error message
271
 * \param[in]    arg        additional error message argument
272
 *                          (will be appended to the error message)
273
 * \param[in]    procname   use __func__
274
 * \param[in]    fval       return float error val
275
 * \return  fval
276
 */
277
l_float32
278
returnErrorFloat1(const char  *msg,
279
                  const char  *arg,
280
                  const char  *procname,
281
                  l_float32    fval)
282
0
{
283
0
    lept_stderr("Leptonica Error in %s: %s: %s\n", procname, msg, arg);
284
0
    return fval;
285
0
}
286
287
288
/*!
289
 * \brief   returnErrorPtr1()
290
 *
291
 * \param[in]    msg        error message
292
 * \param[in]    arg        additional error message argument
293
 *                          (will be appended to the error message)
294
 * \param[in]    procname   use __func__
295
 * \param[in]    pval       return error val (typically null for an error)
296
 * \return  pval
297
 */
298
void *
299
returnErrorPtr1(const char  *msg,
300
                const char  *arg,
301
                const char  *procname,
302
                void        *pval)
303
5.19k
{
304
5.19k
    lept_stderr("Leptonica Error in %s: %s: %s\n", procname, msg, arg);
305
5.19k
    return pval;
306
5.19k
}
307
308
309
/*------------------------------------------------------------------------*
310
 *                   Runtime redirection of stderr                        *
311
 *------------------------------------------------------------------------*
312
 *                                                                        *
313
 *  The user can provide a callback function to redirect messages         *
314
 *  that would otherwise go to stderr.  Here are two examples:            *
315
 *  (1) to stop all messages:                                             *
316
 *      void send_to_devnull(const char *msg) {}                          *
317
 *  (2) to write to the system logger:                                    *
318
 *      void send_to_syslog(const char *msg) {                            *
319
 *           syslog(1, msg);                                              *
320
 *      }                                                                 *
321
 *  These would then be registered using                                  *
322
 *      leptSetStderrHandler(send_to_devnull);                            *
323
 *  and                                                                   *
324
 *      leptSetStderrHandler(send_to_syslog);                             *
325
 *------------------------------------------------------------------------*/
326
    /* By default, all messages go to stderr */
327
static void lept_default_stderr_handler(const char *formatted_msg)
328
0
{
329
0
    if (formatted_msg)
330
0
        fputs(formatted_msg, stderr);
331
0
}
332
333
    /* The stderr callback handler is private to leptonica.
334
     * By default it writes to stderr.  */
335
void (*stderr_handler)(const char *) = lept_default_stderr_handler;
336
337
338
/*!
339
 * \brief   leptSetStderrHandler()
340
 *
341
 * \param[in]    handler   callback function for lept_stderr output
342
 * \return  void
343
 *
344
 * <pre>
345
 * Notes:
346
 *      (1) This registers a handler for redirection of output to stderr
347
 *          at runtime.
348
 *      (2) If called with NULL, the output goes to stderr.
349
 * </pre>
350
 */
351
void leptSetStderrHandler(void (*handler)(const char *))
352
1.90k
{
353
1.90k
    if (handler)
354
1.90k
        stderr_handler = handler;
355
0
    else
356
0
        stderr_handler = lept_default_stderr_handler;
357
1.90k
}
358
359
360
#define MAX_DEBUG_MESSAGE   2000
361
/*!
362
 * \brief   lept_stderr()
363
 *
364
 * \param[in]    fmt      format string
365
 * \param[in]    ...      varargs
366
 * \return  void
367
 *
368
 * <pre>
369
 * Notes:
370
 *      (1) This is a replacement for fprintf(), to allow redirection
371
 *          of output.  All calls to fprintf(stderr, ...) are replaced
372
 *          with calls to lept_stderr(...).
373
 *      (2) The message size is limited to 2K bytes.
374
        (3) This utility was provided by jbarlow83.
375
 * </pre>
376
 */
377
void lept_stderr(const char *fmt, ...)
378
24.2k
{
379
24.2k
va_list  args;
380
24.2k
char     msg[MAX_DEBUG_MESSAGE];
381
24.2k
l_int32  n;
382
383
24.2k
    va_start(args, fmt);
384
24.2k
    n = vsnprintf(msg, sizeof(msg), fmt, args);
385
24.2k
    va_end(args);
386
24.2k
    if (n < 0)
387
0
        return;
388
24.2k
    (*stderr_handler)(msg);
389
24.2k
}
390
391
392
/*--------------------------------------------------------------------*
393
 *                    Test files for equivalence                      *
394
 *--------------------------------------------------------------------*/
395
/*!
396
 * \brief   filesAreIdentical()
397
 *
398
 * \param[in]    fname1
399
 * \param[in]    fname2
400
 * \param[out]   psame     1 if identical; 0 if different
401
 * \return  0 if OK, 1 on error
402
 */
403
l_ok
404
filesAreIdentical(const char  *fname1,
405
                  const char  *fname2,
406
                  l_int32     *psame)
407
0
{
408
0
l_int32   i, same;
409
0
size_t    nbytes1, nbytes2;
410
0
l_uint8  *array1, *array2;
411
412
0
    if (!psame)
413
0
        return ERROR_INT("&same not defined", __func__, 1);
414
0
    *psame = 0;
415
0
    if (!fname1 || !fname2)
416
0
        return ERROR_INT("both names not defined", __func__, 1);
417
418
0
    nbytes1 = nbytesInFile(fname1);
419
0
    nbytes2 = nbytesInFile(fname2);
420
0
    if (nbytes1 != nbytes2)
421
0
        return 0;
422
423
0
    if ((array1 = l_binaryRead(fname1, &nbytes1)) == NULL)
424
0
        return ERROR_INT("array1 not read", __func__, 1);
425
0
    if ((array2 = l_binaryRead(fname2, &nbytes2)) == NULL) {
426
0
        LEPT_FREE(array1);
427
0
        return ERROR_INT("array2 not read", __func__, 1);
428
0
    }
429
0
    same = 1;
430
0
    for (i = 0; i < nbytes1; i++) {
431
0
        if (array1[i] != array2[i]) {
432
0
            same = 0;
433
0
            break;
434
0
        }
435
0
    }
436
0
    LEPT_FREE(array1);
437
0
    LEPT_FREE(array2);
438
0
    *psame = same;
439
440
0
    return 0;
441
0
}
442
443
444
/*--------------------------------------------------------------------------*
445
 *   16 and 32 bit byte-swapping on big endian and little  endian machines  *
446
 *--------------------------------------------------------------------------*
447
 *                                                                          *
448
 *   These are typically used for I/O conversions:                          *
449
 *      (1) endian conversion for data that was read from a file            *
450
 *      (2) endian conversion on data before it is written to a file        *
451
 *--------------------------------------------------------------------------*/
452
453
/*--------------------------------------------------------------------*
454
 *                        16-bit byte swapping                        *
455
 *--------------------------------------------------------------------*/
456
#ifdef L_BIG_ENDIAN
457
458
l_uint16
459
convertOnBigEnd16(l_uint16  shortin)
460
{
461
    return ((shortin << 8) | (shortin >> 8));
462
}
463
464
l_uint16
465
convertOnLittleEnd16(l_uint16  shortin)
466
{
467
    return  shortin;
468
}
469
470
#else     /* L_LITTLE_ENDIAN */
471
472
l_uint16
473
convertOnLittleEnd16(l_uint16  shortin)
474
0
{
475
0
    return ((shortin << 8) | (shortin >> 8));
476
0
}
477
478
l_uint16
479
convertOnBigEnd16(l_uint16  shortin)
480
0
{
481
0
    return  shortin;
482
0
}
483
484
#endif  /* L_BIG_ENDIAN */
485
486
487
/*--------------------------------------------------------------------*
488
 *                        32-bit byte swapping                        *
489
 *--------------------------------------------------------------------*/
490
#ifdef L_BIG_ENDIAN
491
492
l_uint32
493
convertOnBigEnd32(l_uint32  wordin)
494
{
495
    return ((wordin << 24) | ((wordin << 8) & 0x00ff0000) |
496
            ((wordin >> 8) & 0x0000ff00) | (wordin >> 24));
497
}
498
499
l_uint32
500
convertOnLittleEnd32(l_uint32  wordin)
501
{
502
    return wordin;
503
}
504
505
#else  /*  L_LITTLE_ENDIAN */
506
507
l_uint32
508
convertOnLittleEnd32(l_uint32  wordin)
509
0
{
510
0
    return ((wordin << 24) | ((wordin << 8) & 0x00ff0000) |
511
0
            ((wordin >> 8) & 0x0000ff00) | (wordin >> 24));
512
0
}
513
514
l_uint32
515
convertOnBigEnd32(l_uint32  wordin)
516
0
{
517
0
    return wordin;
518
0
}
519
520
#endif  /* L_BIG_ENDIAN */
521
522
523
/*---------------------------------------------------------------------*
524
 *           File corruption and byte replacement operations           *
525
 *---------------------------------------------------------------------*/
526
/*!
527
 * \brief   fileCorruptByDeletion()
528
 *
529
 * \param[in]    filein
530
 * \param[in]    loc       fractional location of start of deletion
531
 * \param[in]    size      fractional size of deletion
532
 * \param[in]    fileout   corrupted file
533
 * \return  0 if OK, 1 on error
534
 *
535
 * <pre>
536
 * Notes:
537
 *      (1) %loc and %size are expressed as a fraction of the file size.
538
 *      (2) This makes a copy of the data in %filein, where bytes in the
539
 *          specified region have deleted.
540
 *      (3) If (%loc + %size) >= 1.0, this deletes from the position
541
 *          represented by %loc to the end of the file.
542
 *      (4) It is useful for testing robustness of I/O wrappers when the
543
 *          data is corrupted, by simulating data corruption by deletion.
544
 * </pre>
545
 */
546
l_ok
547
fileCorruptByDeletion(const char  *filein,
548
                      l_float32    loc,
549
                      l_float32    size,
550
                      const char  *fileout)
551
0
{
552
0
l_int32   i, locb, sizeb, rembytes;
553
0
size_t    inbytes, outbytes;
554
0
l_uint8  *datain, *dataout;
555
556
0
    if (!filein || !fileout)
557
0
        return ERROR_INT("filein and fileout not both specified", __func__, 1);
558
0
    if (loc < 0.0 || loc >= 1.0)
559
0
        return ERROR_INT("loc must be in [0.0 ... 1.0)", __func__, 1);
560
0
    if (size <= 0.0)
561
0
        return ERROR_INT("size must be > 0.0", __func__, 1);
562
0
    if (loc + size > 1.0)
563
0
        size = 1.0f - loc;
564
565
0
    datain = l_binaryRead(filein, &inbytes);
566
0
    locb = (l_int32)(loc * inbytes + 0.5);
567
0
    locb = L_MIN(locb, inbytes - 1);
568
0
    sizeb = (l_int32)(size * inbytes + 0.5);
569
0
    sizeb = L_MAX(1, sizeb);
570
0
    sizeb = L_MIN(sizeb, inbytes - locb);  /* >= 1 */
571
0
    L_INFO("Removed %d bytes at location %d\n", __func__, sizeb, locb);
572
0
    rembytes = inbytes - locb - sizeb;  /* >= 0; to be copied, after excision */
573
574
0
    outbytes = inbytes - sizeb;
575
0
    dataout = (l_uint8 *)LEPT_CALLOC(outbytes, 1);
576
0
    for (i = 0; i < locb; i++)
577
0
        dataout[i] = datain[i];
578
0
    for (i = 0; i < rembytes; i++)
579
0
        dataout[locb + i] = datain[locb + sizeb + i];
580
0
    l_binaryWrite(fileout, "w", dataout, outbytes);
581
582
0
    LEPT_FREE(datain);
583
0
    LEPT_FREE(dataout);
584
0
    return 0;
585
0
}
586
587
588
/*!
589
 * \brief   fileCorruptByMutation()
590
 *
591
 * \param[in]    filein
592
 * \param[in]    loc       fractional location of start of randomization
593
 * \param[in]    size      fractional size of randomization
594
 * \param[in]    fileout   corrupted file
595
 * \return  0 if OK, 1 on error
596
 *
597
 * <pre>
598
 * Notes:
599
 *      (1) %loc and %size are expressed as a fraction of the file size.
600
 *      (2) This makes a copy of the data in %filein, where bytes in the
601
 *          specified region have been replaced by random data.
602
 *      (3) If (%loc + %size) >= 1.0, this modifies data from the position
603
 *          represented by %loc to the end of the file.
604
 *      (4) It is useful for testing robustness of I/O wrappers when the
605
 *          data is corrupted, by simulating data corruption.
606
 * </pre>
607
 */
608
l_ok
609
fileCorruptByMutation(const char  *filein,
610
                      l_float32    loc,
611
                      l_float32    size,
612
                      const char  *fileout)
613
0
{
614
0
l_int32   i, locb, sizeb;
615
0
size_t    bytes;
616
0
l_uint8  *data;
617
618
0
    if (!filein || !fileout)
619
0
        return ERROR_INT("filein and fileout not both specified", __func__, 1);
620
0
    if (loc < 0.0 || loc >= 1.0)
621
0
        return ERROR_INT("loc must be in [0.0 ... 1.0)", __func__, 1);
622
0
    if (size <= 0.0)
623
0
        return ERROR_INT("size must be > 0.0", __func__, 1);
624
0
    if (loc + size > 1.0)
625
0
        size = 1.0f - loc;
626
627
0
    data = l_binaryRead(filein, &bytes);
628
0
    locb = (l_int32)(loc * bytes + 0.5);
629
0
    locb = L_MIN(locb, bytes - 1);
630
0
    sizeb = (l_int32)(size * bytes + 0.5);
631
0
    sizeb = L_MAX(1, sizeb);
632
0
    sizeb = L_MIN(sizeb, bytes - locb);  /* >= 1 */
633
0
    L_INFO("Randomizing %d bytes at location %d\n", __func__, sizeb, locb);
634
635
        /* Make an array of random bytes and do the substitution */
636
0
    for (i = 0; i < sizeb; i++) {
637
0
        data[locb + i] =
638
0
            (l_uint8)(255.9 * ((l_float64)rand() / (l_float64)RAND_MAX));
639
0
    }
640
641
0
    l_binaryWrite(fileout, "w", data, bytes);
642
0
    LEPT_FREE(data);
643
0
    return 0;
644
0
}
645
646
647
/*!
648
 * \brief   fileReplaceBytes()
649
 *
650
 * \param[in]    filein      input file
651
 * \param[in]    start       start location for replacement
652
 * \param[in]    nbytes      number of bytes to be removed
653
 * \param[in]    newdata     replacement bytes
654
 * \param[in]    newsize     size of replacement bytes
655
 * \param[in]    fileout     output file
656
 * \return  0 if OK, 1 on error
657
 *
658
 * <pre>
659
 * Notes:
660
 *      (1) To remove %nbytes without replacement, set %newdata == NULL.
661
 *      (2) One use is for replacing the date/time in a pdf file by a
662
 *          string of 12 '0's, effectively removing the date without
663
 *          invalidating the byte counters in the pdf file:
664
 *              fileReplaceBytes(filein 86 12 (char *)"000000000000" 12 fileout
665
 * </pre>
666
 */
667
l_ok
668
fileReplaceBytes(const char  *filein,
669
                 l_int32      start,
670
                 l_int32      nbytes,
671
                 l_uint8     *newdata,
672
                 size_t       newsize,
673
                 const char  *fileout)
674
0
{
675
0
l_int32   i, index;
676
0
size_t    inbytes, outbytes;
677
0
l_uint8  *datain, *dataout;
678
679
0
    if (!filein || !fileout)
680
0
        return ERROR_INT("filein and fileout not both specified", __func__, 1);
681
682
0
    datain = l_binaryRead(filein, &inbytes);
683
0
    if (start + nbytes > inbytes)
684
0
        L_WARNING("start + nbytes > length(filein) = %zu\n", __func__, inbytes);
685
686
0
    if (!newdata) newsize = 0;
687
0
    outbytes = inbytes - nbytes + newsize;
688
0
    if ((dataout = (l_uint8 *)LEPT_CALLOC(outbytes, 1)) == NULL) {
689
0
        LEPT_FREE(datain);
690
0
        return ERROR_INT("calloc fail for dataout", __func__, 1);
691
0
    }
692
693
0
    for (i = 0; i < start; i++)
694
0
        dataout[i] = datain[i];
695
0
    for (i = start; i < start + newsize; i++)
696
0
        dataout[i] = newdata[i - start];
697
0
    index = start + nbytes;  /* for datain */
698
0
    start += newsize;  /* for dataout */
699
0
    for (i = start; i < outbytes; i++, index++)
700
0
        dataout[i] = datain[index];
701
0
    l_binaryWrite(fileout, "w", dataout, outbytes);
702
703
0
    LEPT_FREE(datain);
704
0
    LEPT_FREE(dataout);
705
0
    return 0;
706
0
}
707
708
709
/*---------------------------------------------------------------------*
710
 *              Generate random integer in given interval              *
711
 *---------------------------------------------------------------------*/
712
/*!
713
 * \brief   genRandomIntOnInterval()
714
 *
715
 * \param[in]    start     beginning of interval; can be < 0
716
 * \param[in]    end       end of interval; must be >= start
717
 * \param[in]    seed      use 0 to skip; otherwise call srand
718
 * \param[out]   pval      random integer in interval [start ... end]
719
 * \return  0 if OK, 1 on error
720
 */
721
l_ok
722
genRandomIntOnInterval(l_int32   start,
723
                       l_int32   end,
724
                       l_int32   seed,
725
                       l_int32  *pval)
726
0
{
727
0
l_float64  range;
728
729
0
    if (!pval)
730
0
        return ERROR_INT("&val not defined", __func__, 1);
731
0
    *pval = 0;
732
0
    if (end < start)
733
0
        return ERROR_INT("invalid range", __func__, 1);
734
735
0
    if (seed > 0) srand(seed);
736
0
    range = (l_float64)(end - start + 1);
737
0
    *pval = start + (l_int32)((l_float64)range *
738
0
                       ((l_float64)rand() / (l_float64)RAND_MAX));
739
0
    return 0;
740
0
}
741
742
743
/*---------------------------------------------------------------------*
744
 *                        Simple math functions                        *
745
 *---------------------------------------------------------------------*/
746
/*!
747
 * \brief   lept_roundftoi()
748
 *
749
 * \param[in]    fval
750
 * \return  value rounded to the nearest integer
751
 *
752
 * <pre>
753
 * Notes:
754
 *      (1) For fval >= 0, fval --> round(fval) == floor(fval + 0.5)
755
 *          For fval < 0, fval --> -round(-fval))
756
 *          This is symmetric around 0.
757
 *          e.g., for fval in (-0.5 ... 0.5), fval --> 0
758
 * </pre>
759
 */
760
l_int32
761
lept_roundftoi(l_float32  fval)
762
0
{
763
0
    return (fval >= 0.0) ? (l_int32)(fval + 0.5) : (l_int32)(fval - 0.5);
764
0
}
765
766
767
/*!
768
 * \brief   lept_floor()
769
 *
770
 * \param[in]    fval
771
 * \return  largest integer that is not greater than %fval
772
 */
773
l_int32
774
lept_floor(l_float32  fval)
775
0
{
776
0
    return (fval >= 0.0) ? (l_int32)(fval) : -(l_int32)(-fval);
777
0
}
778
779
780
/*!
781
 * \brief   lept_ceiling()
782
 *
783
 * \param[in]    fval
784
 * \return  smallest integer that is not less than %fval
785
 *
786
 * <pre>
787
 * Notes:
788
 *      (1) If fval is equal to its interger value, return that.
789
 *          Otherwise:
790
 *            For fval > 0, fval --> 1 + floor(fval)
791
 *            For fval < 0, fval --> -(1 + floor(-fval))
792
 * </pre>
793
 */
794
l_int32
795
lept_ceiling(l_float32  fval)
796
0
{
797
0
    return (fval == (l_int32)fval ? (l_int32)fval :
798
0
            fval > 0.0 ? 1 + (l_int32)(fval) : -(1 + (l_int32)(-fval)));
799
0
}
800
801
802
/*---------------------------------------------------------------------*
803
 *                        64-bit hash functions                        *
804
 *---------------------------------------------------------------------*/
805
/*!
806
 * \brief   l_hashStringToUint64()
807
 *
808
 * \param[in]    str
809
 * \param[out]   phash    hash value
810
 * \return  0 if OK, 1 on error
811
 *
812
 * <pre>
813
 * Notes:
814
 *      (1) The intent of the hash is to avoid collisions by mapping
815
 *          the string as randomly as possible into 64 bits.
816
 *      (2) To the extent that the hashes are random, the probability of
817
 *          a collision can be approximated by the square of the number
818
 *          of strings divided by 2^64.  For 1 million strings, the
819
 *          collision probability is about 1 in 16 million.
820
 *      (3) I expect non-randomness of the distribution to be most evident
821
 *          for small text strings.  This hash function has been tested
822
 *          for all 5-character text strings composed of 26 letters,
823
 *          of which there are 26^5 = 12356630.  There are no hash
824
 *          collisions for this set.
825
 * </pre>
826
 */
827
l_ok
828
l_hashStringToUint64(const char  *str,
829
                     l_uint64    *phash)
830
0
{
831
0
l_uint64  hash, mulp;
832
833
0
    if (phash) *phash = 0;
834
0
    if (!str || (str[0] == '\0'))
835
0
        return ERROR_INT("str not defined or empty", __func__, 1);
836
0
    if (!phash)
837
0
        return ERROR_INT("&hash not defined", __func__, 1);
838
839
0
    mulp = 26544357894361247;  /* prime, about 1/700 of the max uint64 */
840
0
    hash = 104395301;
841
0
    while (*str) {
842
0
        hash += (*str++ * mulp) ^ (hash >> 7);   /* shift [1...23] are ok */
843
0
    }
844
0
    *phash = hash ^ (hash << 37);
845
0
    return 0;
846
0
}
847
848
849
/*!
850
 * \brief   l_hashStringToUint64Fast()
851
 *
852
 * \param[in]    str
853
 * \param[out]   phash    hash value
854
 * \return  0 if OK, 1 on error
855
 *
856
 * <pre>
857
 * Notes:
858
 *     (1) This very simple hash algorithm is described in "The Practice
859
 *         of Programming" by Kernighan and Pike, p. 57 (1999).
860
 *     (2) The returned hash value would then be hashed into an index into
861
 *         the hashtable, using the mod operator with the hashtable size.
862
 * </pre>
863
 */
864
l_ok
865
l_hashStringToUint64Fast(const char  *str,
866
                         l_uint64    *phash)
867
0
{
868
0
l_uint64  h;
869
0
l_uint8  *p;
870
871
0
    if (phash) *phash = 0;
872
0
    if (!str || (str[0] == '\0'))
873
0
        return ERROR_INT("str not defined or empty", __func__, 1);
874
0
    if (!phash)
875
0
        return ERROR_INT("&hash not defined", __func__, 1);
876
877
0
    h = 0;
878
0
    for (p = (l_uint8 *)str; *p != '\0'; p++)
879
0
        h = 37 * h + *p;  /* 37 is good prime number for this */
880
0
    *phash = h;
881
0
    return 0;
882
0
}
883
884
885
/*!
886
 * \brief   l_hashPtToUint64()
887
 *
888
 * \param[in]    x, y
889
 * \param[out]   phash    hash value
890
 * \return  0 if OK, 1 on error
891
 *
892
 * <pre>
893
 * Notes:
894
 *      (1) This simple hash function has no collisions for
895
 *          any of 400 million points with x and y up to 20000.
896
 * </pre>
897
 */
898
l_ok
899
l_hashPtToUint64(l_int32    x,
900
                 l_int32    y,
901
                 l_uint64  *phash)
902
0
{
903
0
    if (!phash)
904
0
        return ERROR_INT("&hash not defined", __func__, 1);
905
906
0
    *phash = (l_uint64)(2173249142.3849 * x + 3763193258.6227 * y);
907
0
    return 0;
908
0
}
909
910
911
/*!
912
 * \brief   l_hashFloat64ToUint64()
913
 *
914
 * \param[in]    val
915
 * \param[out]   phash      hash key
916
 * \return  0 if OK, 1 on error
917
 *
918
 * <pre>
919
 * Notes:
920
 *      (1) This is a simple hash for using hashmaps with 64-bit float data.
921
 *      (2) The resulting hash is called a "key" in a lookup operation.
922
 *          The bucket for %val in a hashmap is then found by taking the mod
923
 *          of the hash key with the number of buckets (which is prime).
924
 * </pre>
925
 */
926
l_ok
927
l_hashFloat64ToUint64(l_float64  val,
928
                      l_uint64  *phash)
929
0
{
930
0
    if (!phash)
931
0
        return ERROR_INT("&hash not defined", __func__, 1);
932
0
    val = (val >= 0.0) ? 847019.66701 * val : -217324.91613 * val;
933
0
    *phash = (l_uint64)val;
934
0
    return 0;
935
0
}
936
937
938
/*---------------------------------------------------------------------*
939
 *                           Prime finders                             *
940
 *---------------------------------------------------------------------*/
941
/*!
942
 * \brief   findNextLargerPrime()
943
 *
944
 * \param[in]    start
945
 * \param[out]   pprime    first prime larger than %start
946
 * \return  0 if OK, 1 on error
947
 */
948
l_ok
949
findNextLargerPrime(l_int32    start,
950
                    l_uint32  *pprime)
951
0
{
952
0
l_int32  i, is_prime;
953
954
0
    if (!pprime)
955
0
        return ERROR_INT("&prime not defined", __func__, 1);
956
0
    *pprime = 0;
957
0
    if (start <= 0)
958
0
        return ERROR_INT("start must be > 0", __func__, 1);
959
960
0
    for (i = start + 1; ; i++) {
961
0
        lept_isPrime(i, &is_prime, NULL);
962
0
        if (is_prime) {
963
0
            *pprime = i;
964
0
            return 0;
965
0
        }
966
0
    }
967
968
0
    return ERROR_INT("prime not found!", __func__, 1);
969
0
}
970
971
972
/*!
973
 * \brief   lept_isPrime()
974
 *
975
 * \param[in]    n           64-bit unsigned
976
 * \param[out]   pis_prime   1 if prime, 0 otherwise
977
 * \param[out]   pfactor     [optional] smallest divisor, or 0 on error
978
 *                           or if prime
979
 * \return  0 if OK, 1 on error
980
 */
981
l_ok
982
lept_isPrime(l_uint64   n,
983
             l_int32   *pis_prime,
984
             l_uint32  *pfactor)
985
0
{
986
0
l_uint32  div;
987
0
l_uint64  limit, ratio;
988
989
0
    if (pis_prime) *pis_prime = 0;
990
0
    if (pfactor) *pfactor = 0;
991
0
    if (!pis_prime)
992
0
        return ERROR_INT("&is_prime not defined", __func__, 1);
993
0
    if (n <= 0)
994
0
        return ERROR_INT("n must be > 0", __func__, 1);
995
996
0
    if (n % 2 == 0) {
997
0
        if (pfactor) *pfactor = 2;
998
0
        return 0;
999
0
    }
1000
1001
0
    limit = (l_uint64)sqrt((l_float64)n);
1002
0
    for (div = 3; div < limit; div += 2) {
1003
0
       ratio = n / div;
1004
0
       if (ratio * div == n) {
1005
0
           if (pfactor) *pfactor = div;
1006
0
           return 0;
1007
0
       }
1008
0
    }
1009
1010
0
    *pis_prime = 1;
1011
0
    return 0;
1012
0
}
1013
1014
1015
/*---------------------------------------------------------------------*
1016
 *                         Gray code conversion                        *
1017
 *---------------------------------------------------------------------*/
1018
/*!
1019
 * \brief   convertIntToGrayCode()
1020
 *
1021
 * \param[in]  val    integer value
1022
 * \return     corresponding gray code value
1023
 *
1024
 * <pre>
1025
 * Notes:
1026
 *      (1) Gray code values corresponding to integers differ by
1027
 *          only one bit transition between successive integers.
1028
 * </pre>
1029
 */
1030
l_uint32
1031
convertIntToGrayCode(l_uint32 val)
1032
0
{
1033
0
    return (val >> 1) ^ val;
1034
0
}
1035
1036
1037
/*!
1038
 * \brief   convertGrayCodeToInt()
1039
 *
1040
 * \param[in]  val    gray code value
1041
 * \return     corresponding integer value
1042
 */
1043
l_uint32
1044
convertGrayCodeToInt(l_uint32 val)
1045
0
{
1046
0
l_uint32  shift;
1047
1048
0
    for (shift = 1; shift < 32; shift <<= 1)
1049
0
        val ^= val >> shift;
1050
0
    return val;
1051
0
}
1052
1053
1054
/*---------------------------------------------------------------------*
1055
 *                       Leptonica version number                      *
1056
 *---------------------------------------------------------------------*/
1057
/*!
1058
 * \brief   getLeptonicaVersion()
1059
 *
1060
 *      Return: string of version number (e.g., 'leptonica-1.74.2')
1061
 *
1062
 *  Notes:
1063
 *      (1) The caller has responsibility to free the memory.
1064
 */
1065
char *
1066
getLeptonicaVersion(void)
1067
0
{
1068
0
size_t  bufsize = 100;
1069
1070
0
    char *version = (char *)LEPT_CALLOC(bufsize, sizeof(char));
1071
1072
#ifdef _MSC_VER
1073
  #ifdef _USRDLL
1074
    char dllStr[] = "DLL";
1075
  #else
1076
    char dllStr[] = "LIB";
1077
  #endif
1078
  #ifdef _DEBUG
1079
    char debugStr[] = "Debug";
1080
  #else
1081
    char debugStr[] = "Release";
1082
  #endif
1083
  #ifdef _M_IX86
1084
    char bitStr[] = " x86";
1085
  #elif _M_X64
1086
    char bitStr[] = " x64";
1087
  #else
1088
    char bitStr[] = "";
1089
  #endif
1090
    snprintf(version, bufsize, "leptonica-%d.%d.%d (%s, %s) [MSC v.%d %s %s%s]",
1091
           LIBLEPT_MAJOR_VERSION, LIBLEPT_MINOR_VERSION, LIBLEPT_PATCH_VERSION,
1092
           __DATE__, __TIME__, _MSC_VER, dllStr, debugStr, bitStr);
1093
1094
#else
1095
1096
0
    snprintf(version, bufsize, "leptonica-%d.%d.%d", LIBLEPT_MAJOR_VERSION,
1097
0
             LIBLEPT_MINOR_VERSION, LIBLEPT_PATCH_VERSION);
1098
1099
0
#endif   /* _MSC_VER */
1100
0
    return version;
1101
0
}
1102
1103
1104
/*---------------------------------------------------------------------*
1105
 *                           Timing procs                              *
1106
 *---------------------------------------------------------------------*/
1107
#if !defined(_WIN32) && !defined(__Fuchsia__)
1108
1109
#include <sys/time.h>
1110
#include <sys/resource.h>
1111
1112
static struct rusage rusage_before;
1113
static struct rusage rusage_after;
1114
1115
/*!
1116
 * \brief   startTimer(), stopTimer()
1117
 *
1118
 *  Notes:
1119
 *      (1) These measure the cpu time elapsed between the two calls:
1120
 *            startTimer();
1121
 *            ....
1122
 *            lept_stderr( "Elapsed time = %7.3f sec\n", stopTimer());
1123
 */
1124
void
1125
startTimer(void)
1126
0
{
1127
0
    getrusage(RUSAGE_SELF, &rusage_before);
1128
0
}
1129
1130
l_float32
1131
stopTimer(void)
1132
0
{
1133
0
l_int32  tsec, tusec;
1134
1135
0
    getrusage(RUSAGE_SELF, &rusage_after);
1136
1137
0
    tsec = rusage_after.ru_utime.tv_sec - rusage_before.ru_utime.tv_sec;
1138
0
    tusec = rusage_after.ru_utime.tv_usec - rusage_before.ru_utime.tv_usec;
1139
0
    return (tsec + ((l_float32)tusec) / 1000000.0);
1140
0
}
1141
1142
1143
/*!
1144
 * \brief   startTimerNested(), stopTimerNested()
1145
 *
1146
 *  Example of usage:
1147
 *
1148
 *      L_TIMER  t1 = startTimerNested();
1149
 *      ....
1150
 *      L_TIMER  t2 = startTimerNested();
1151
 *      ....
1152
 *      lept_stderr( "Elapsed time 2 = %7.3f sec\n", stopTimerNested(t2));
1153
 *      ....
1154
 *      lept_stderr( "Elapsed time 1 = %7.3f sec\n", stopTimerNested(t1));
1155
 */
1156
L_TIMER
1157
startTimerNested(void)
1158
0
{
1159
0
struct rusage  *rusage_start;
1160
1161
0
    rusage_start = (struct rusage *)LEPT_CALLOC(1, sizeof(struct rusage));
1162
0
    getrusage(RUSAGE_SELF, rusage_start);
1163
0
    return rusage_start;
1164
0
}
1165
1166
l_float32
1167
stopTimerNested(L_TIMER  rusage_start)
1168
0
{
1169
0
l_int32        tsec, tusec;
1170
0
struct rusage  rusage_stop;
1171
1172
0
    getrusage(RUSAGE_SELF, &rusage_stop);
1173
1174
0
    tsec = rusage_stop.ru_utime.tv_sec -
1175
0
           ((struct rusage *)rusage_start)->ru_utime.tv_sec;
1176
0
    tusec = rusage_stop.ru_utime.tv_usec -
1177
0
           ((struct rusage *)rusage_start)->ru_utime.tv_usec;
1178
0
    LEPT_FREE(rusage_start);
1179
0
    return (tsec + ((l_float32)tusec) / 1000000.0);
1180
0
}
1181
1182
1183
/*!
1184
 * \brief   l_getCurrentTime()
1185
 *
1186
 * \param[out]   sec     [optional] in seconds since birth of Unix
1187
 * \param[out]   usec    [optional] in microseconds since birth of Unix
1188
 * \return  void
1189
 */
1190
void
1191
l_getCurrentTime(l_int32  *sec,
1192
                 l_int32  *usec)
1193
0
{
1194
0
struct timeval tv;
1195
1196
0
    gettimeofday(&tv, NULL);
1197
0
    if (sec) *sec = (l_int32)tv.tv_sec;
1198
0
    if (usec) *usec = (l_int32)tv.tv_usec;
1199
0
}
1200
1201
#elif defined(__Fuchsia__) /* resource.h not implemented on Fuchsia. */
1202
1203
    /* Timer functions are used for testing and debugging, and
1204
     * are stubbed out.  If they are needed in the future, they
1205
     * can be implemented in Fuchsia using the zircon syscall
1206
     * zx_object_get_info() in ZX_INFOR_THREAD_STATS mode.  */
1207
void
1208
startTimer(void)
1209
{
1210
}
1211
1212
l_float32
1213
stopTimer(void)
1214
{
1215
    return 0.0;
1216
}
1217
1218
L_TIMER
1219
startTimerNested(void)
1220
{
1221
    return NULL;
1222
}
1223
1224
l_float32
1225
stopTimerNested(L_TIMER  rusage_start)
1226
{
1227
    return 0.0;
1228
}
1229
1230
void
1231
l_getCurrentTime(l_int32  *sec,
1232
                 l_int32  *usec)
1233
{
1234
}
1235
1236
#else   /* _WIN32 : resource.h not implemented under Windows */
1237
1238
    /* Note: if division by 10^7 seems strange, the time is expressed
1239
     * as the number of 100-nanosecond intervals that have elapsed
1240
     * since 12:00 A.M. January 1, 1601.  */
1241
1242
static ULARGE_INTEGER utime_before;
1243
static ULARGE_INTEGER utime_after;
1244
1245
void
1246
startTimer(void)
1247
{
1248
HANDLE    this_process;
1249
FILETIME  start, stop, kernel, user;
1250
1251
    this_process = GetCurrentProcess();
1252
1253
    GetProcessTimes(this_process, &start, &stop, &kernel, &user);
1254
1255
    utime_before.LowPart  = user.dwLowDateTime;
1256
    utime_before.HighPart = user.dwHighDateTime;
1257
}
1258
1259
l_float32
1260
stopTimer(void)
1261
{
1262
HANDLE     this_process;
1263
FILETIME   start, stop, kernel, user;
1264
ULONGLONG  hnsec;  /* in units of hecto-nanosecond (100 ns) intervals */
1265
1266
    this_process = GetCurrentProcess();
1267
1268
    GetProcessTimes(this_process, &start, &stop, &kernel, &user);
1269
1270
    utime_after.LowPart  = user.dwLowDateTime;
1271
    utime_after.HighPart = user.dwHighDateTime;
1272
    hnsec = utime_after.QuadPart - utime_before.QuadPart;
1273
    return (l_float32)(signed)hnsec / 10000000.0f;
1274
}
1275
1276
L_TIMER
1277
startTimerNested(void)
1278
{
1279
HANDLE           this_process;
1280
FILETIME         start, stop, kernel, user;
1281
ULARGE_INTEGER  *utime_start;
1282
1283
    this_process = GetCurrentProcess();
1284
1285
    GetProcessTimes (this_process, &start, &stop, &kernel, &user);
1286
1287
    utime_start = (ULARGE_INTEGER *)LEPT_CALLOC(1, sizeof(ULARGE_INTEGER));
1288
    utime_start->LowPart  = user.dwLowDateTime;
1289
    utime_start->HighPart = user.dwHighDateTime;
1290
    return utime_start;
1291
}
1292
1293
l_float32
1294
stopTimerNested(L_TIMER  utime_start)
1295
{
1296
HANDLE          this_process;
1297
FILETIME        start, stop, kernel, user;
1298
ULARGE_INTEGER  utime_stop;
1299
ULONGLONG       hnsec;  /* in units of 100 ns intervals */
1300
1301
    this_process = GetCurrentProcess ();
1302
1303
    GetProcessTimes (this_process, &start, &stop, &kernel, &user);
1304
1305
    utime_stop.LowPart  = user.dwLowDateTime;
1306
    utime_stop.HighPart = user.dwHighDateTime;
1307
    hnsec = utime_stop.QuadPart - ((ULARGE_INTEGER *)utime_start)->QuadPart;
1308
    LEPT_FREE(utime_start);
1309
    return (l_float32)(signed)hnsec / 10000000.0f;
1310
}
1311
1312
void
1313
l_getCurrentTime(l_int32  *sec,
1314
                 l_int32  *usec)
1315
{
1316
ULARGE_INTEGER  utime, birthunix;
1317
FILETIME        systemtime;
1318
LONGLONG        birthunixhnsec = 116444736000000000;  /*in units of 100 ns */
1319
LONGLONG        usecs;
1320
1321
    GetSystemTimeAsFileTime(&systemtime);
1322
    utime.LowPart  = systemtime.dwLowDateTime;
1323
    utime.HighPart = systemtime.dwHighDateTime;
1324
1325
    birthunix.LowPart = (DWORD) birthunixhnsec;
1326
    birthunix.HighPart = birthunixhnsec >> 32;
1327
1328
    usecs = (LONGLONG) ((utime.QuadPart - birthunix.QuadPart) / 10);
1329
1330
    if (sec) *sec = (l_int32) (usecs / 1000000);
1331
    if (usec) *usec = (l_int32) (usecs % 1000000);
1332
}
1333
1334
#endif
1335
1336
1337
/*!
1338
 * \brief   startWallTimer()
1339
 *
1340
 * \return  walltimer-ptr
1341
 *
1342
 * <pre>
1343
 * Notes:
1344
 *      (1) These measure the wall clock time  elapsed between the two calls:
1345
 *            L_WALLTIMER *timer = startWallTimer();
1346
 *            ....
1347
 *            lept_stderr( "Elapsed time = %f sec\n", stopWallTimer(&timer);
1348
 *      (2) Note that the timer object is destroyed by stopWallTimer().
1349
 * </pre>
1350
 */
1351
L_WALLTIMER *
1352
startWallTimer(void)
1353
0
{
1354
0
L_WALLTIMER  *timer;
1355
1356
0
    timer = (L_WALLTIMER *)LEPT_CALLOC(1, sizeof(L_WALLTIMER));
1357
0
    l_getCurrentTime(&timer->start_sec, &timer->start_usec);
1358
0
    return timer;
1359
0
}
1360
1361
/*!
1362
 * \brief   stopWallTimer()
1363
 *
1364
 * \param[in,out]  ptimer     walltimer pointer
1365
 * \return  time wall time elapsed in seconds
1366
 */
1367
l_float32
1368
stopWallTimer(L_WALLTIMER  **ptimer)
1369
0
{
1370
0
l_int32       tsec, tusec;
1371
0
L_WALLTIMER  *timer;
1372
1373
0
    if (!ptimer)
1374
0
        return (l_float32)ERROR_FLOAT("&timer not defined", __func__, 0.0);
1375
0
    timer = *ptimer;
1376
0
    if (!timer)
1377
0
        return (l_float32)ERROR_FLOAT("timer not defined", __func__, 0.0);
1378
1379
0
    l_getCurrentTime(&timer->stop_sec, &timer->stop_usec);
1380
0
    tsec = timer->stop_sec - timer->start_sec;
1381
0
    tusec = timer->stop_usec - timer->start_usec;
1382
0
    LEPT_FREE(timer);
1383
0
    *ptimer = NULL;
1384
0
    return (tsec + ((l_float32)tusec) / 1000000.0f);
1385
0
}
1386
1387
1388
/*!
1389
 * \brief   l_getFormattedDate()
1390
 *
1391
 * \return  formatted date string, or NULL on error
1392
 *
1393
 * <pre>
1394
 * Notes:
1395
 *      (1) This is used in pdf, in the form specified in section 3.8.2 of
1396
 *          http://partners.adobe.com/public/developer/en/pdf/PDFReference.pdf
1397
 *      (2) Contributed by Dave Bryan.  Works on all platforms.
1398
 * </pre>
1399
 */
1400
char *
1401
l_getFormattedDate(void)
1402
0
{
1403
0
char        buf[128] = "", sep = 'Z';
1404
0
l_int32     gmt_offset, relh, relm;
1405
0
time_t      ut, lt;
1406
0
struct tm   Tm;
1407
0
struct tm  *tptr = &Tm;
1408
1409
0
    ut = time(NULL);
1410
1411
        /* This generates a second "time_t" value by calling "gmtime" to
1412
           fill in a "tm" structure expressed as UTC and then calling
1413
           "mktime", which expects a "tm" structure expressed as the
1414
           local time.  The result is a value that is offset from the
1415
           value returned by the "time" function by the local UTC offset.
1416
           "tm_isdst" is set to -1 to tell "mktime" to determine for
1417
           itself whether DST is in effect.  This is necessary because
1418
           "gmtime" always sets "tm_isdst" to 0, which would tell
1419
           "mktime" to presume that DST is not in effect. */
1420
#ifdef _WIN32
1421
  #ifdef _MSC_VER
1422
    gmtime_s(tptr, &ut);
1423
  #else  /* mingw */
1424
    tptr = gmtime(&ut);
1425
  #endif
1426
#else
1427
0
    gmtime_r(&ut, tptr);
1428
0
#endif
1429
0
    tptr->tm_isdst = -1;
1430
0
    lt = mktime(tptr);
1431
1432
        /* Calls "difftime" to obtain the resulting difference in seconds,
1433
         * because "time_t" is an opaque type, per the C standard. */
1434
0
    gmt_offset = (l_int32) difftime(ut, lt);
1435
0
    if (gmt_offset > 0)
1436
0
        sep = '+';
1437
0
    else if (gmt_offset < 0)
1438
0
        sep = '-';
1439
0
    relh = L_ABS(gmt_offset) / 3600;
1440
0
    relm = (L_ABS(gmt_offset) % 3600) / 60;
1441
1442
#ifdef _WIN32
1443
  #ifdef _MSC_VER
1444
    localtime_s(tptr, &ut);
1445
  #else  /* mingw */
1446
    tptr = localtime(&ut);
1447
  #endif
1448
#else
1449
0
    localtime_r(&ut, tptr);
1450
0
#endif
1451
0
    strftime(buf, sizeof(buf), "%Y%m%d%H%M%S", tptr);
1452
0
    sprintf(buf + 14, "%c%02d'%02d'", sep, relh, relm);
1453
0
    return stringNew(buf);
1454
0
}