droid48/jni/timer.c
2011-06-28 12:42:30 +02:00

722 lines
16 KiB
C

/*
* This file is part of x48, an emulator of the HP-48sx Calculator.
* Copyright (C) 1994 Eddie C. Dost (ecd@dressler.de)
*
* This program is free software; you can redistribute it and/or modify
* it under the terms of the GNU General Public License as published by
* the Free Software Foundation; either version 2 of the License, or
* (at your option) any later version.
*
* This program is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
* GNU General Public License for more details.
*
* You should have received a copy of the GNU General Public License
* along with this program; if not, write to the Free Software
* Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA.
*/
/* $Log: timer.c,v $
* Revision 1.7 1995/01/11 18:20:01 ecd
* major update to support HP48 G/GX
*
* Revision 1.6 1994/12/07 20:20:50 ecd
* minor fix
*
* Revision 1.6 1994/12/07 20:20:50 ecd
* minor fix
*
* Revision 1.5 1994/11/28 02:00:51 ecd
* removed stupid bug that caused negative time on call
* to adjtime()
*
* Revision 1.4 1994/11/02 14:44:28 ecd
* real time support completed
*
*
* $Id: timer.c,v 1.7 1995/01/11 18:20:01 ecd Exp ecd $
*/
#include "global.h"
#include <stdlib.h>
#include <stdio.h>
#include <unistd.h>
#include <time.h>
#include "timer.h"
#include "debugger.h"
#include "romio.h"
/* #define DEBUG_TIMER 1 */
/* #define DEBUG_TIMER_ADJUST 1 */
#ifdef SOLARIS
extern int gettimeofday __ProtoType__((struct timeval *tp));
#endif
#ifdef SUNOS
extern int gettimeofday __ProtoType__((struct timeval *, struct timezone *));
#endif
typedef struct x48_timer_t {
word_1 run;
word_64 start;
word_64 stop;
word_64 value;
} x48_timer_t;
static x48_timer_t timers[NR_TIMERS];
static long systime_offset = 0;
/*
* Ticks for THU 01.01.1970 00:00:00
*/
word_64 unix_0_time = { 0x0001cf2e, 0x8f800000 };
word_64 ticks_10_min = { 0x0, 0x00b40000 };
/*
* Will be in saturn_t in the future
*/
word_64 set_0_time = { 0x0, 0x0 };
/*
* Calculated as (unix_0_time + set_0_time)
*/
word_64 time_offset = { 0x0, 0x0 };
#define RAM_BASE_SX 0x70000
#define ACCESSTIME_SX (0x70052 - RAM_BASE_SX)
#define ACCESSCRC_SX (0x7005F - RAM_BASE_SX)
#define TIMEOUT_SX (0x70063 - RAM_BASE_SX)
#define TIMEOUTCLK_SX (0x70070 - RAM_BASE_SX)
#define RAM_BASE_GX 0x80000
#define ACCESSTIME_GX (0x80058 - RAM_BASE_GX)
#define ACCESSCRC_GX (0x80065 - RAM_BASE_GX)
#define TIMEOUT_GX (0x80069 - RAM_BASE_GX)
#define TIMEOUTCLK_GX (0x80076 - RAM_BASE_GX)
#define calc_crc(nib) (crc = (crc >> 4) ^ (((crc ^ (nib)) & 0xf) * 0x1081))
static inline void
#ifdef __FunctionProto__
add_64(word_64 r1, word_64 *r2)
#else
add_64(r1, r2)
word_64 r1;
word_64 *r2;
#endif
{
unsigned short s1[4], s2[4];
unsigned long r[4];
s1[0] = (unsigned short)(r1.lo & 0xffff);
s1[1] = (unsigned short)((r1.lo >> 16) & 0xffff);
s1[2] = (unsigned short)(r1.hi & 0xffff);
s1[3] = (unsigned short)((r1.hi >> 16) & 0xffff);
s2[0] = (unsigned short)(r2->lo & 0xffff);
s2[1] = (unsigned short)((r2->lo >> 16) & 0xffff);
s2[2] = (unsigned short)(r2->hi & 0xffff);
s2[3] = (unsigned short)((r2->hi >> 16) & 0xffff);
r[0] = (unsigned long)s1[0] + (unsigned long)s2[0];
r[1] = (unsigned long)s1[1] + (unsigned long)s2[1];
r[2] = (unsigned long)s1[2] + (unsigned long)s2[2];
r[3] = (unsigned long)s1[3] + (unsigned long)s2[3];
if (r[0] >> 16) r[1]++;
if (r[1] >> 16) r[2]++;
if (r[2] >> 16) r[3]++;
r2->lo = ((r[1] << 16) | (r[0] & 0xffff));
r2->hi = ((r[3] << 16) | (r[2] & 0xffff));
}
static inline void
#ifdef __FunctionProto__
sub_64(word_64 r1, word_64 *r2)
#else
sub_64(r1, r2)
word_64 r1;
word_64 *r2;
#endif
{
unsigned short s1[4], s2[4];
unsigned long r[4];
s1[0] = (unsigned short)(r1.lo & 0xffff);
s1[1] = (unsigned short)((r1.lo >> 16) & 0xffff);
s1[2] = (unsigned short)(r1.hi & 0xffff);
s1[3] = (unsigned short)((r1.hi >> 16) & 0xffff);
s2[0] = (unsigned short)(r2->lo & 0xffff);
s2[1] = (unsigned short)((r2->lo >> 16) & 0xffff);
s2[2] = (unsigned short)(r2->hi & 0xffff);
s2[3] = (unsigned short)((r2->hi >> 16) & 0xffff);
r[0] = (unsigned long)s2[0] - (unsigned long)s1[0];
r[1] = (unsigned long)s2[1] - (unsigned long)s1[1];
r[2] = (unsigned long)s2[2] - (unsigned long)s1[2];
r[3] = (unsigned long)s2[3] - (unsigned long)s1[3];
if (r[0] >> 16) r[1]--;
if (r[1] >> 16) r[2]--;
if (r[2] >> 16) r[3]--;
r2->lo = ((r[1] << 16) | (r[0] & 0xffff));
r2->hi = ((r[3] << 16) | (r[2] & 0xffff));
}
/*
* Set ACCESSTIME: (on startup)
*
* 1. TICKS = 8192 * gettimeofday() (UNIX System Time)
* 2. TICKS += unix_0_time (TICKS for 1.1.1970, 0:00)
* 3. TICKS += set_0_time (Time adjustment from User)
* 4. TICKS += saturn.timer2 (Timer 2 from last run)
* 5. Write this into ACCESSTIME
* 6. Calculate CRC for 13 Nibbles
* 7. Write this into ACCESSCRC
* 8. Prevent AutoOff by setting TIMEOUT
*
*/
void
#ifdef __FunctionProto__
set_accesstime(void)
#else
set_accesstime()
#endif
{
struct timeval tv;
#ifndef SOLARIS
struct timezone tz;
#endif
word_64 ticks, timeout, timer2;
word_20 accesstime_loc, timeout_loc;
word_20 accesscrc_loc, timeoutclk_loc;
word_16 crc;
word_4 val;
int i;
time_t gmt;
struct tm *ltm;
/*
* This is done to set the variable 'timezone' on SYSV systems
*/
(void)time(&gmt);
ltm = localtime(&gmt);
#ifdef SYSV_TIME
systime_offset = timezone;
if( ltm->tm_isdst )
systime_offset -= 3600;
#else
systime_offset = -ltm->tm_gmtoff;
#endif
#ifdef SOLARIS
gettimeofday(&tv);
#else
gettimeofday(&tv, &tz);
#endif
tv.tv_sec -= systime_offset;
ticks.hi = (tv.tv_sec >> 19);
ticks.lo = (tv.tv_sec << 13);
ticks.lo |= (tv.tv_usec << 7) / 15625;
time_offset.lo = unix_0_time.lo;
time_offset.hi = unix_0_time.hi;
add_64(set_0_time, &time_offset);
add_64(time_offset, &ticks);
timer2.lo = saturn.timer2;
if (saturn.timer2 & 0x80000000)
timer2.hi = 0xffffffff;
else
timer2.hi = 0x0;
add_64(timer2, &ticks);
timeout.lo = ticks.lo;
timeout.hi = ticks.hi;
crc = 0x0;
if (opt_gx)
{
accesstime_loc = ACCESSTIME_GX;
accesscrc_loc = ACCESSCRC_GX;
timeout_loc = TIMEOUT_GX;
timeoutclk_loc = TIMEOUTCLK_GX;
}
else
{
accesstime_loc = ACCESSTIME_SX;
accesscrc_loc = ACCESSCRC_SX;
timeout_loc = TIMEOUT_SX;
timeoutclk_loc = TIMEOUTCLK_SX;
}
for (i = 0; i < 13; i++)
{
val = ticks.lo & 0xf;
calc_crc(val);
saturn.ram[accesstime_loc + i] = val;
ticks.lo >>= 4;
ticks.lo |= (ticks.hi & 0xf) << 28;
ticks.hi >>= 4;
}
for (i = 0; i < 4; i++)
{
saturn.ram[accesscrc_loc + i] = crc & 0xf;
crc >>= 4;
}
add_64(ticks_10_min, &timeout);
for (i = 0; i < 13; i++)
{
val = timeout.lo & 0xf;
calc_crc(val);
saturn.ram[timeout_loc + i] = val;
timeout.lo >>= 4;
timeout.lo |= (timeout.hi & 0xf) << 28;
timeout.hi >>= 4;
}
saturn.ram[timeoutclk_loc] = 0xf;
}
long
#ifdef __FunctionProto__
diff_timer(word_64 *t1, word_64 *t2)
#else
diff_timer(t1, t2)
word_64 *t1;
word_64 *t2;
#endif
{
return (t1->lo - t2->lo);
}
static inline void
#ifdef __FunctionProto__
add_sub_64(word_64 *t1, word_64 *t2, word_64 *diff)
#else
add_sub_64(t1, t2, diff)
word_64 *t1;
word_64 *t2;
word_64 *diff;
#endif
{
unsigned short s1[4], s2[4];
unsigned long r[4];
s1[0] = (unsigned short)(t1->lo & 0xffff);
s1[1] = (unsigned short)((t1->lo >> 16) & 0xffff);
s1[2] = (unsigned short)(t1->hi & 0xffff);
s1[3] = (unsigned short)((t1->hi >> 16) & 0xffff);
s2[0] = (unsigned short)(t2->lo & 0xffff);
s2[1] = (unsigned short)((t2->lo >> 16) & 0xffff);
s2[2] = (unsigned short)(t2->hi & 0xffff);
s2[3] = (unsigned short)((t2->hi >> 16) & 0xffff);
r[0] = (unsigned long)s1[0] - (unsigned long)s2[0];
r[1] = (unsigned long)s1[1] - (unsigned long)s2[1];
r[2] = (unsigned long)s1[2] - (unsigned long)s2[2];
r[3] = (unsigned long)s1[3] - (unsigned long)s2[3];
if (r[0] >> 16) r[1]--;
if (r[1] >> 16) r[2]--;
if (r[2] >> 16) r[3]--;
s1[0] = (unsigned short)(diff->lo & 0xffff);
s1[1] = (unsigned short)((diff->lo >> 16) & 0xffff);
s1[2] = (unsigned short)(diff->hi & 0xffff);
s1[3] = (unsigned short)((diff->hi >> 16) & 0xffff);
s2[0] = (unsigned short)r[0];
s2[1] = (unsigned short)r[1];
s2[2] = (unsigned short)r[2];
s2[3] = (unsigned short)r[3];
r[0] = (unsigned long)s1[0] + (unsigned long)s2[0];
r[1] = (unsigned long)s1[1] + (unsigned long)s2[1];
r[2] = (unsigned long)s1[2] + (unsigned long)s2[2];
r[3] = (unsigned long)s1[3] + (unsigned long)s2[3];
if (r[0] >> 16) r[1]++;
if (r[1] >> 16) r[2]++;
if (r[2] >> 16) r[3]++;
t2->lo = t1->lo;
t2->hi = t1->hi;
diff->lo = ((r[1] << 16) | (r[0] & 0xffff));
diff->hi = ((r[3] << 16) | (r[2] & 0xffff));
}
void
#ifdef __FunctionProto__
start_timer(int timer)
#else
start_timer(timer)
int timer;
#endif
{
struct timeval tv;
#ifndef SOLARIS
struct timezone tz;
#endif
if (timer > NR_TIMERS)
return;
if (timers[timer].run == 1)
return;
#ifdef SOLARIS
gettimeofday(&tv);
#else
gettimeofday(&tv, &tz);
#endif
tv.tv_sec -= systime_offset;
timers[timer].run = 1;
if (timer == T1_TIMER) {
timers[timer].start.hi = (tv.tv_sec >> 23);
timers[timer].start.lo = (tv.tv_sec << 9);
timers[timer].start.lo |= tv.tv_usec / 62500;
} else {
timers[timer].start.hi = (tv.tv_sec >> 19);
timers[timer].start.lo = (tv.tv_sec << 13);
timers[timer].start.lo |= (tv.tv_usec << 7) / 15625;
}
#ifdef DEBUG_TIMER
fprintf(stderr, "Timer[%d] start at 0x%.8lx%.8lx\n", timer,
timers[timer].start.hi, timers[timer].start.lo);
#endif
}
void
#ifdef __FunctionProto__
restart_timer(int timer)
#else
restart_timer(timer)
int timer;
#endif
{
struct timeval tv;
#ifndef SOLARIS
struct timezone tz;
#endif
if (timer > NR_TIMERS)
return;
timers[timer].start.lo = timers[timer].start.hi = 0;
timers[timer].stop.lo = timers[timer].stop.hi = 0;
timers[timer].value.lo = timers[timer].value.hi = 0;
#ifdef SOLARIS
gettimeofday(&tv);
#else
gettimeofday(&tv, &tz);
#endif
tv.tv_sec -= systime_offset;
timers[timer].run = 1;
if (timer == T1_TIMER) {
timers[timer].start.hi = (tv.tv_sec >> 23);
timers[timer].start.lo = (tv.tv_sec << 9);
timers[timer].start.lo |= tv.tv_usec / 62500;
} else {
timers[timer].start.hi = (tv.tv_sec >> 19);
timers[timer].start.lo = (tv.tv_sec << 13);
timers[timer].start.lo |= (tv.tv_usec << 7) / 15625;
}
#ifdef DEBUG_TIMER
fprintf(stderr, "Timer[%d] restart at 0x%.8lx%.8lx\n", timer,
timers[timer].start.hi, timers[timer].start.lo);
#endif
}
void
#ifdef __FunctionProto__
stop_timer(int timer)
#else
stop_timer(timer)
int timer;
#endif
{
struct timeval tv;
#ifndef SOLARIS
struct timezone tz;
#endif
if (timer > NR_TIMERS)
return;
if (timers[timer].run == 0)
return;
#ifdef SOLARIS
gettimeofday(&tv);
#else
gettimeofday(&tv, &tz);
#endif
tv.tv_sec -= systime_offset;
timers[timer].run = 0;
if (timer == T1_TIMER) {
timers[timer].stop.hi = (tv.tv_sec >> 23);
timers[timer].stop.lo = (tv.tv_sec << 9);
timers[timer].stop.lo |= tv.tv_usec / 62500;
} else {
timers[timer].stop.hi = (tv.tv_sec >> 19);
timers[timer].stop.lo = (tv.tv_sec << 13);
timers[timer].stop.lo |= (tv.tv_usec << 7) / 15625;
}
add_sub_64(&timers[timer].stop, &timers[timer].start, &timers[timer].value);
#ifdef DEBUG_TIMER
fprintf(stderr, "Timer[%d] stop at 0x%.8lx%.8lx, value 0x%.8lx%.8lx\n",
timer, timers[timer].stop.hi, timers[timer].stop.lo,
timers[timer].value.hi, timers[timer].value.lo);
#endif
}
void
#ifdef __FunctionProto__
reset_timer(int timer)
#else
reset_timer(timer)
int timer;
#endif
{
if (timer > NR_TIMERS)
return;
timers[timer].run = 0;
timers[timer].start.lo = timers[timer].start.hi = 0;
timers[timer].stop.lo = timers[timer].stop.hi = 0;
timers[timer].value.lo = timers[timer].value.hi = 0;
#ifdef DEBUG_TIMER
fprintf(stderr, "Timer[%d] reset\n", timer);
#endif
}
static word_64 zero = { 0, 0 };
word_64
#ifdef __FunctionProto__
get_timer(int timer)
#else
get_timer(timer)
int timer;
#endif
{
struct timeval tv;
#ifndef SOLARIS
struct timezone tz;
#endif
word_64 stop;
if (timer > NR_TIMERS)
return zero;
if (timers[timer].run) {
#ifdef SOLARIS
gettimeofday(&tv);
#else
gettimeofday(&tv, &tz);
#endif
tv.tv_sec -= systime_offset;
if (timer == T1_TIMER) {
stop.hi = (tv.tv_sec >> 23);
stop.lo = (tv.tv_sec << 9);
stop.lo |= tv.tv_usec / 62500;
} else {
stop.hi = (tv.tv_sec >> 19);
stop.lo = (tv.tv_sec << 13);
stop.lo |= (tv.tv_usec << 7) / 15625;
}
add_sub_64(&stop, &timers[timer].start, &timers[timer].value);
}
return timers[timer].value;
}
/*
* Calculate TIMER 2 Ticks:
*
* 1. TICKS = 8192 * gettimeofday() (UNIX System Time)
* 2. TICKS += unix_0_time (TICKS for 1.1.1970, 0:00)
* 3. TICKS += set_0_time (Time adjustment from User)
* 4. Get value of ACCESSTIME
* 5. Return (ACCESSTIME - TICKS)
*
*/
t1_t2_ticks
#ifdef __FunctionProto__
get_t1_t2(void)
#else
get_t1_t2()
#endif
{
struct timeval tv;
#ifndef SOLARIS
struct timezone tz;
#endif
word_64 stop;
t1_t2_ticks ticks;
word_64 access_time;
word_64 adj_time;
word_64 diff_time;
word_64 delta;
word_20 accesstime_loc;
int i;
#ifdef SOLARIS
gettimeofday(&tv);
#else
gettimeofday(&tv, &tz);
#endif
tv.tv_sec -= systime_offset;
if (timers[T1_TIMER].run)
{
stop.hi = (tv.tv_sec >> 23);
stop.lo = (tv.tv_sec << 9);
stop.lo |= tv.tv_usec / 62500;
add_sub_64(&stop, &timers[T1_TIMER].start, &timers[T1_TIMER].value);
}
ticks.t1_ticks = timers[T1_TIMER].value.lo;
stop.hi = (tv.tv_sec >> 19);
stop.lo = (tv.tv_sec << 13);
stop.lo |= (tv.tv_usec << 7) / 15625;
add_64(time_offset, &stop);
if (opt_gx)
accesstime_loc = ACCESSTIME_GX;
else
accesstime_loc = ACCESSTIME_SX;
access_time.lo = 0x0;
access_time.hi = 0x0;
for (i = 13 - 1; i >= 0; i--)
{
access_time.hi <<= 4;
access_time.hi |= ((access_time.lo >> 28) & 0xf);
access_time.lo <<= 4;
access_time.lo |= ((int)saturn.ram[accesstime_loc + i] & 0xf);
}
sub_64(stop, &access_time);
if (adj_time_pending || in_debugger)
{
/*
* We have been inside an interrupt for very long, maybe
* or we are sleeping in the debugger.
* Don't adjust the time, can't come from user, anyhow.
*/
if ((saturn.timer2 >= 0 && (access_time.lo & 0x80000000))
|| ((unsigned long)saturn.timer2 > access_time.lo))
{
/*
* check OK, return calculated time
*/
ticks.t2_ticks = access_time.lo;
}
else
{
/*
* Don't increment timer2, return old value and
* slow down timer2.
*/
ticks.t2_ticks = saturn.timer2;
saturn.t2_tick++;
}
return ticks;
}
diff_time.lo = saturn.timer2;
if (saturn.timer2 & 0x80000000)
diff_time.hi = 0xffffffff;
else
diff_time.hi = 0x0;
adj_time.lo = access_time.lo;
adj_time.hi = access_time.hi;
sub_64(diff_time, &adj_time);
if (adj_time.hi & 0x8000000)
{
delta.lo = 0x0;
delta.hi = 0x0;
sub_64(adj_time, &delta);
}
else
{
delta.lo = adj_time.lo;
delta.hi = adj_time.hi;
}
if (delta.hi != 0 || (delta.lo > 0x3c000)) /* Half a minute */
{
add_64(adj_time, &set_0_time);
add_64(adj_time, &time_offset);
sub_64(adj_time, &access_time);
#ifdef DEBUG_TIMER_ADJUST
fprintf(stderr, "Time adjusted by ");
if (adj_time.hi)
fprintf(stderr, "%lX%.8lX", adj_time.hi, adj_time.lo);
else
fprintf(stderr, "%lX", adj_time.lo);
fprintf(stderr, " TICKS, Total offset ");
if (set_0_time.hi)
fprintf(stderr, "%lX%.8lX", set_0_time.hi, set_0_time.lo);
else
fprintf(stderr, "%lX", set_0_time.lo);
fprintf(stderr, " TICKS\n");
#endif
}
if ((saturn.timer2 >= 0 && (access_time.lo & 0x80000000))
|| ((unsigned long)saturn.timer2 > access_time.lo))
{
/*
* check OK, return calculated time
*/
ticks.t2_ticks = access_time.lo;
}
else
{
/*
* Don't increment timer2, return old value and
* slow down timer2.
*/
ticks.t2_ticks = saturn.timer2;
saturn.t2_tick++;
}
return ticks;
}