mirror of
https://github.com/holub/mame
synced 2025-05-06 14:25:54 +03:00

* New abbreviated types are in osd and util namespaces, and also in global namespace for things that #include "emu.h" * Get rid of import of cstdint types to global namespace (C99 does this anyway) * Remove the cstdint types from everything in emu * Get rid of U64/S64 macros * Fix a bug in dps16 caused by incorrect use of macro * Fix debugcon not checking for "do " prefix case-insensitively * Fix a lot of messed up tabulation * More constexpr * Fix up many __names
157 lines
4.4 KiB
C++
157 lines
4.4 KiB
C++
// license:BSD-3-Clause
|
|
// copyright-holders:Aaron Giles
|
|
/***************************************************************************
|
|
|
|
attotime.c
|
|
|
|
Support functions for working with attotime data.
|
|
|
|
***************************************************************************/
|
|
|
|
#include "emucore.h"
|
|
#include "eminline.h"
|
|
#include "attotime.h"
|
|
|
|
|
|
//**************************************************************************
|
|
// GLOBAL VARIABLES
|
|
//**************************************************************************
|
|
|
|
const attotime attotime::zero(0, 0);
|
|
const attotime attotime::never(ATTOTIME_MAX_SECONDS, 0);
|
|
|
|
//**************************************************************************
|
|
// CORE MATH FUNCTIONS
|
|
//**************************************************************************
|
|
|
|
//-------------------------------------------------
|
|
// operator*= - multiply an attotime by a
|
|
// constant
|
|
//-------------------------------------------------
|
|
|
|
attotime &attotime::operator*=(u32 factor)
|
|
{
|
|
// if one of the items is attotime::never, return attotime::never
|
|
if (m_seconds >= ATTOTIME_MAX_SECONDS)
|
|
return *this = never;
|
|
|
|
// 0 times anything is zero
|
|
if (factor == 0)
|
|
return *this = zero;
|
|
|
|
// split attoseconds into upper and lower halves which fit into 32 bits
|
|
u32 attolo;
|
|
u32 attohi = divu_64x32_rem(m_attoseconds, ATTOSECONDS_PER_SECOND_SQRT, &attolo);
|
|
|
|
// scale the lower half, then split into high/low parts
|
|
u64 temp = mulu_32x32(attolo, factor);
|
|
u32 reslo;
|
|
temp = divu_64x32_rem(temp, ATTOSECONDS_PER_SECOND_SQRT, &reslo);
|
|
|
|
// scale the upper half, then split into high/low parts
|
|
temp += mulu_32x32(attohi, factor);
|
|
u32 reshi;
|
|
temp = divu_64x32_rem(temp, ATTOSECONDS_PER_SECOND_SQRT, &reshi);
|
|
|
|
// scale the seconds
|
|
temp += mulu_32x32(m_seconds, factor);
|
|
if (temp >= ATTOTIME_MAX_SECONDS)
|
|
return *this = never;
|
|
|
|
// build the result
|
|
m_seconds = temp;
|
|
m_attoseconds = (attoseconds_t)reslo + mul_32x32(reshi, ATTOSECONDS_PER_SECOND_SQRT);
|
|
return *this;
|
|
}
|
|
|
|
|
|
//-------------------------------------------------
|
|
// operator/= - divide an attotime by a constant
|
|
//-------------------------------------------------
|
|
|
|
attotime &attotime::operator/=(u32 factor)
|
|
{
|
|
// if one of the items is attotime::never, return attotime::never
|
|
if (m_seconds >= ATTOTIME_MAX_SECONDS)
|
|
return *this = never;
|
|
|
|
// ignore divide by zero
|
|
if (factor == 0)
|
|
return *this;
|
|
|
|
// split attoseconds into upper and lower halves which fit into 32 bits
|
|
u32 attolo;
|
|
u32 attohi = divu_64x32_rem(m_attoseconds, ATTOSECONDS_PER_SECOND_SQRT, &attolo);
|
|
|
|
// divide the seconds and get the remainder
|
|
u32 remainder;
|
|
m_seconds = divu_64x32_rem(m_seconds, factor, &remainder);
|
|
|
|
// combine the upper half of attoseconds with the remainder and divide that
|
|
u64 temp = s64(attohi) + mulu_32x32(remainder, ATTOSECONDS_PER_SECOND_SQRT);
|
|
u32 reshi = divu_64x32_rem(temp, factor, &remainder);
|
|
|
|
// combine the lower half of attoseconds with the remainder and divide that
|
|
temp = attolo + mulu_32x32(remainder, ATTOSECONDS_PER_SECOND_SQRT);
|
|
u32 reslo = divu_64x32_rem(temp, factor, &remainder);
|
|
|
|
// round based on the remainder
|
|
m_attoseconds = (attoseconds_t)reslo + mulu_32x32(reshi, ATTOSECONDS_PER_SECOND_SQRT);
|
|
if (remainder >= factor / 2)
|
|
if (++m_attoseconds >= ATTOSECONDS_PER_SECOND)
|
|
{
|
|
m_attoseconds = 0;
|
|
m_seconds++;
|
|
}
|
|
return *this;
|
|
}
|
|
|
|
|
|
//-------------------------------------------------
|
|
// as_string - return a temporary printable
|
|
// string describing an attotime
|
|
//-------------------------------------------------
|
|
|
|
const char *attotime::as_string(int precision) const
|
|
{
|
|
static char buffers[8][30];
|
|
static int nextbuf;
|
|
char *buffer = &buffers[nextbuf++ % 8][0];
|
|
|
|
// special case: never
|
|
if (*this == never)
|
|
sprintf(buffer, "%-*s", precision, "(never)");
|
|
|
|
// case 1: we want no precision; seconds only
|
|
else if (precision == 0)
|
|
sprintf(buffer, "%d", m_seconds);
|
|
|
|
// case 2: we want 9 or fewer digits of precision
|
|
else if (precision <= 9)
|
|
{
|
|
u32 upper = m_attoseconds / ATTOSECONDS_PER_SECOND_SQRT;
|
|
int temp = precision;
|
|
while (temp < 9)
|
|
{
|
|
upper /= 10;
|
|
temp++;
|
|
}
|
|
sprintf(buffer, "%d.%0*d", m_seconds, precision, upper);
|
|
}
|
|
|
|
// case 3: more than 9 digits of precision
|
|
else
|
|
{
|
|
u32 lower;
|
|
u32 upper = divu_64x32_rem(m_attoseconds, ATTOSECONDS_PER_SECOND_SQRT, &lower);
|
|
int temp = precision;
|
|
while (temp < 18)
|
|
{
|
|
lower /= 10;
|
|
temp++;
|
|
}
|
|
sprintf(buffer, "%d.%09d%0*d", m_seconds, upper, precision - 9, lower);
|
|
}
|
|
return buffer;
|
|
}
|