[PATCH] _mktm_r: optimize speed
Freddie Chopin
freddie_chopin@op.pl
Wed Sep 3 17:30:00 GMT 2014
I attach patch (with ChangeLog entry) for _mktm_r() which replaces
iterative calculations of day with a few divisions and multiplications.
This change results in the function working two times faster on ARM
Cortex-M3 microcontroller - one million calls of "old" function using
random numbers take ~70s, while the same calculations with "new" code
take ~33s (clock of chip is 16MHz). Obviously the improvement depends on
the input value - sometimes the patched code can be slower than the old
one - for input values "close" to year 1970. On the other hand, the
"further" from 1970, the bigger is the performance gain.
Both versions give the same results - tested with a few million random
numbers - both versions were called for the same input value and result
was compared with memcmp().
Regards,
FCh
-------------- next part --------------
2014-09-03 Freddie Chopin <freddie_chopin@op.pl>
* libc/time/mktm_r.c (_mktm_r): Optimize speed.
-------------- next part --------------
From 1179360279c5ba9b9c1bf53ccfa0b46f1bbdfc61 Mon Sep 17 00:00:00 2001
From: Freddie Chopin <freddie.chopin@gmail.com>
Date: Wed, 3 Sep 2014 19:16:12 +0200
Subject: [PATCH] _mktm_r: optimize speed
---
newlib/libc/time/mktm_r.c | 49 +++++++++++++++++++++++++++++++----------------
1 file changed, 33 insertions(+), 16 deletions(-)
diff --git a/newlib/libc/time/mktm_r.c b/newlib/libc/time/mktm_r.c
index 9a3bc82..debf2c3 100644
--- a/newlib/libc/time/mktm_r.c
+++ b/newlib/libc/time/mktm_r.c
@@ -14,14 +14,23 @@
#include <time.h>
#include "local.h"
+/* there are 97 leap years in 400-year periods */
+#define DAYS_PER_400_YEARS ((400 - 97) * 365 + 97 * 366)
+/* there are 24 leap years in 100-year periods */
+#define DAYS_PER_100_YEARS ((100 - 24) * 365 + 24 * 366)
+/* there is one leap year every 4 years */
+#define DAYS_PER_4_YEARS (3 * 365 + 366)
+
static _CONST int mon_lengths[2][MONSPERYEAR] = {
{31, 28, 31, 30, 31, 30, 31, 31, 30, 31, 30, 31},
{31, 29, 31, 30, 31, 30, 31, 31, 30, 31, 30, 31}
} ;
-static _CONST int year_lengths[2] = {
- 365,
- 366
+static _CONST int days_per_year[4] = {
+ 365, /* 1970 or 1966 */
+ 365, /* 1971 or 1967 */
+ 366, /* 1972 or 1968 */
+ 365 /* 1973 or 1969 */
} ;
struct tm *
@@ -64,30 +73,38 @@ _DEFUN (_mktm_r, (tim_p, res, is_gmtime),
res->tm_wday += DAYSPERWEEK;
/* compute year & day of year */
- y = EPOCH_YEAR;
- if (days >= 0)
+ years400 = days / DAYS_PER_400_YEARS;
+ days -= years400 * DAYS_PER_400_YEARS;
+ years100 = days / DAYS_PER_100_YEARS;
+ days -= years100 * DAYS_PER_100_YEARS;
+ years4 = days / DAYS_PER_4_YEARS;
+ days -= years4 * DAYS_PER_4_YEARS;
+
+ y = EPOCH_YEAR + years400 * 400 + years100 * 100 + years4 * 4;
+
+ /* the value left in days is based in 1970 */
+ if (days < 0)
{
- for (;;)
+ ip = &days_per_year[3];
+ while (days < 0)
{
- yleap = isleap(y);
- if (days < year_lengths[yleap])
- break;
- y++;
- days -= year_lengths[yleap];
+ days += *ip--;
+ --y;
}
}
else
{
- do
+ ip = &days_per_year[0];
+ while (days >= *ip)
{
- --y;
- yleap = isleap(y);
- days += year_lengths[yleap];
- } while (days < 0);
+ days -= *ip++;
+ ++y;
+ }
}
res->tm_year = y - YEAR_BASE;
res->tm_yday = days;
+ yleap = isleap(y);
ip = mon_lengths[yleap];
for (res->tm_mon = 0; days >= ip[res->tm_mon]; ++res->tm_mon)
days -= ip[res->tm_mon];
--
1.8.1.msysgit.1
More information about the Newlib
mailing list