mirror of
https://github.com/holub/mame
synced 2025-06-01 02:21:48 +03:00

Remove redundant machine items from address_space and device_t. Neither machine nor m_machine are directly accessible anymore. Instead a new getter machine() is available which returns a machine reference. So: space->machine->xxx ==> space->machine().xxx device->machine->yyy ==> device->machine().yyy Globally changed all running_machine pointers to running_machine references. Any function/method that takes a running_machine takes it as a required parameter (1 or 2 exceptions). Being consistent here gets rid of a lot of odd &machine or *machine, but it does mean a very large bulk change across the project. Structs which have a running_machine * now have that variable renamed to m_machine, and now have a shiny new machine() method that works like the space and device methods above. Since most of these are things that should eventually be devices anyway, consider this a step in that direction. 98% of the update was done with regex searches. The changes are architected such that the compiler will catch the remaining errors: // find things that use an embedded machine directly and replace // with a machine() getter call S: ->machine-> R: ->machine\(\)\. // do the same if via a reference S: \.machine-> R: \.machine\(\)\. // convert function parameters to running_machine & S: running_machine \*machine([^;]) R: running_machine \&machine\1 // replace machine-> with machine. S: machine-> R: machine\. // replace &machine() with machine() S: \&([()->a-z0-9_]+machine\(\)) R: \1 // sanity check: look for this used as a cast (running_machine &) // and change to this: *(running_machine *)
350 lines
8.3 KiB
C
350 lines
8.3 KiB
C
/*****************************************************************************
|
|
|
|
Harris HC-55516 (and related) emulator
|
|
|
|
Copyright Nicola Salmoria and the MAME Team
|
|
|
|
*****************************************************************************/
|
|
|
|
#include "emu.h"
|
|
#include "hc55516.h"
|
|
|
|
|
|
/* 4x oversampling */
|
|
#define SAMPLE_RATE (48000 * 4)
|
|
|
|
#define INTEGRATOR_LEAK_TC 0.001
|
|
#define FILTER_DECAY_TC 0.004
|
|
#define FILTER_CHARGE_TC 0.004
|
|
#define FILTER_MIN 0.0416
|
|
#define FILTER_MAX 1.0954
|
|
#define SAMPLE_GAIN 10000.0
|
|
|
|
|
|
typedef struct _hc55516_state hc55516_state;
|
|
struct _hc55516_state
|
|
{
|
|
sound_stream *channel;
|
|
int clock; /* 0 = software driven, non-0 = oscillator */
|
|
int active_clock_hi;
|
|
UINT8 shiftreg_mask;
|
|
|
|
UINT8 last_clock_state;
|
|
UINT8 digit;
|
|
UINT8 new_digit;
|
|
UINT8 shiftreg;
|
|
|
|
INT16 curr_sample;
|
|
INT16 next_sample;
|
|
|
|
UINT32 update_count;
|
|
|
|
double filter;
|
|
double integrator;
|
|
};
|
|
|
|
|
|
static double charge, decay, leak;
|
|
|
|
|
|
static STREAM_UPDATE( hc55516_update );
|
|
|
|
|
|
|
|
INLINE hc55516_state *get_safe_token(device_t *device)
|
|
{
|
|
assert(device != NULL);
|
|
assert(device->type() == HC55516 ||
|
|
device->type() == MC3417 ||
|
|
device->type() == MC3418);
|
|
return (hc55516_state *)downcast<legacy_device_base *>(device)->token();
|
|
}
|
|
|
|
|
|
static void start_common(device_t *device, UINT8 _shiftreg_mask, int _active_clock_hi)
|
|
{
|
|
hc55516_state *chip = get_safe_token(device);
|
|
|
|
/* compute the fixed charge, decay, and leak time constants */
|
|
charge = pow(exp(-1.0), 1.0 / (FILTER_CHARGE_TC * 16000.0));
|
|
decay = pow(exp(-1.0), 1.0 / (FILTER_DECAY_TC * 16000.0));
|
|
leak = pow(exp(-1.0), 1.0 / (INTEGRATOR_LEAK_TC * 16000.0));
|
|
|
|
chip->clock = device->clock();
|
|
chip->shiftreg_mask = _shiftreg_mask;
|
|
chip->active_clock_hi = _active_clock_hi;
|
|
chip->last_clock_state = 0;
|
|
|
|
/* create the stream */
|
|
chip->channel = device->machine().sound().stream_alloc(*device, 0, 1, SAMPLE_RATE, chip, hc55516_update);
|
|
|
|
device->save_item(NAME(chip->last_clock_state));
|
|
device->save_item(NAME(chip->digit));
|
|
device->save_item(NAME(chip->new_digit));
|
|
device->save_item(NAME(chip->shiftreg));
|
|
device->save_item(NAME(chip->curr_sample));
|
|
device->save_item(NAME(chip->next_sample));
|
|
device->save_item(NAME(chip->update_count));
|
|
device->save_item(NAME(chip->filter));
|
|
device->save_item(NAME(chip->integrator));
|
|
}
|
|
|
|
|
|
static DEVICE_START( hc55516 )
|
|
{
|
|
start_common(device, 0x07, TRUE);
|
|
}
|
|
|
|
|
|
static DEVICE_START( mc3417 )
|
|
{
|
|
start_common(device, 0x07, FALSE);
|
|
}
|
|
|
|
|
|
static DEVICE_START( mc3418 )
|
|
{
|
|
start_common(device, 0x0f, FALSE);
|
|
}
|
|
|
|
|
|
|
|
static DEVICE_RESET( hc55516 )
|
|
{
|
|
hc55516_state *chip = get_safe_token(device);
|
|
chip->last_clock_state = 0;
|
|
}
|
|
|
|
|
|
|
|
INLINE int is_external_osciallator(hc55516_state *chip)
|
|
{
|
|
return chip->clock != 0;
|
|
}
|
|
|
|
|
|
INLINE int is_active_clock_transition(hc55516_state *chip, int clock_state)
|
|
{
|
|
return (( chip->active_clock_hi && !chip->last_clock_state && clock_state) ||
|
|
(!chip->active_clock_hi && chip->last_clock_state && !clock_state));
|
|
}
|
|
|
|
|
|
INLINE int current_clock_state(hc55516_state *chip)
|
|
{
|
|
return ((UINT64)chip->update_count * chip->clock * 2 / SAMPLE_RATE) & 0x01;
|
|
}
|
|
|
|
|
|
static void process_digit(hc55516_state *chip)
|
|
{
|
|
double integrator = chip->integrator, temp;
|
|
|
|
/* shift the bit into the shift register */
|
|
chip->shiftreg = (chip->shiftreg << 1) | chip->digit;
|
|
|
|
/* move the estimator up or down a step based on the bit */
|
|
if (chip->digit)
|
|
integrator += chip->filter;
|
|
else
|
|
integrator -= chip->filter;
|
|
|
|
/* simulate leakage */
|
|
integrator *= leak;
|
|
|
|
/* if we got all 0's or all 1's in the last n bits, bump the step up */
|
|
if (((chip->shiftreg & chip->shiftreg_mask) == 0) ||
|
|
((chip->shiftreg & chip->shiftreg_mask) == chip->shiftreg_mask))
|
|
{
|
|
chip->filter = FILTER_MAX - ((FILTER_MAX - chip->filter) * charge);
|
|
|
|
if (chip->filter > FILTER_MAX)
|
|
chip->filter = FILTER_MAX;
|
|
}
|
|
|
|
/* simulate decay */
|
|
else
|
|
{
|
|
chip->filter *= decay;
|
|
|
|
if (chip->filter < FILTER_MIN)
|
|
chip->filter = FILTER_MIN;
|
|
}
|
|
|
|
/* compute the sample as a 32-bit word */
|
|
temp = integrator * SAMPLE_GAIN;
|
|
chip->integrator = integrator;
|
|
|
|
/* compress the sample range to fit better in a 16-bit word */
|
|
if (temp < 0)
|
|
chip->next_sample = (int)(temp / (-temp * (1.0 / 32768.0) + 1.0));
|
|
else
|
|
chip->next_sample = (int)(temp / (temp * (1.0 / 32768.0) + 1.0));
|
|
}
|
|
|
|
|
|
static STREAM_UPDATE( hc55516_update )
|
|
{
|
|
hc55516_state *chip = (hc55516_state *)param;
|
|
stream_sample_t *buffer = outputs[0];
|
|
int i;
|
|
INT32 sample, slope;
|
|
|
|
/* zero-length? bail */
|
|
if (samples == 0)
|
|
return;
|
|
|
|
if (!is_external_osciallator(chip))
|
|
{
|
|
/* track how many samples we've updated without a clock */
|
|
chip->update_count += samples;
|
|
if (chip->update_count > SAMPLE_RATE / 32)
|
|
{
|
|
chip->update_count = SAMPLE_RATE;
|
|
chip->next_sample = 0;
|
|
}
|
|
}
|
|
|
|
/* compute the interpolation slope */
|
|
sample = chip->curr_sample;
|
|
slope = ((INT32)chip->next_sample - sample) / samples;
|
|
chip->curr_sample = chip->next_sample;
|
|
|
|
if (is_external_osciallator(chip))
|
|
{
|
|
/* external oscillator */
|
|
for (i = 0; i < samples; i++, sample += slope)
|
|
{
|
|
UINT8 clock_state;
|
|
|
|
*buffer++ = sample;
|
|
|
|
chip->update_count++;
|
|
|
|
clock_state = current_clock_state(chip);
|
|
|
|
/* pull in next digit on the appropriate edge of the clock */
|
|
if (is_active_clock_transition(chip, clock_state))
|
|
{
|
|
chip->digit = chip->new_digit;
|
|
|
|
process_digit(chip);
|
|
}
|
|
|
|
chip->last_clock_state = clock_state;
|
|
}
|
|
}
|
|
|
|
/* software driven clock */
|
|
else
|
|
for (i = 0; i < samples; i++, sample += slope)
|
|
*buffer++ = sample;
|
|
}
|
|
|
|
|
|
void hc55516_clock_w(device_t *device, int state)
|
|
{
|
|
hc55516_state *chip = get_safe_token(device);
|
|
UINT8 clock_state = state ? TRUE : FALSE;
|
|
|
|
/* only makes sense for setups with a software driven clock */
|
|
assert(!is_external_osciallator(chip));
|
|
|
|
/* speech clock changing? */
|
|
if (is_active_clock_transition(chip, clock_state))
|
|
{
|
|
/* update the output buffer before changing the registers */
|
|
chip->channel->update();
|
|
|
|
/* clear the update count */
|
|
chip->update_count = 0;
|
|
|
|
process_digit(chip);
|
|
}
|
|
|
|
/* update the clock */
|
|
chip->last_clock_state = clock_state;
|
|
}
|
|
|
|
|
|
void hc55516_digit_w(device_t *device, int digit)
|
|
{
|
|
hc55516_state *chip = get_safe_token(device);
|
|
|
|
if (is_external_osciallator(chip))
|
|
{
|
|
chip->channel->update();
|
|
chip->new_digit = digit & 1;
|
|
}
|
|
else
|
|
chip->digit = digit & 1;
|
|
}
|
|
|
|
|
|
int hc55516_clock_state_r(device_t *device)
|
|
{
|
|
hc55516_state *chip = get_safe_token(device);
|
|
|
|
/* only makes sense for setups with an external oscillator */
|
|
assert(is_external_osciallator(chip));
|
|
|
|
chip->channel->update();
|
|
|
|
return current_clock_state(chip);
|
|
}
|
|
|
|
|
|
|
|
/**************************************************************************
|
|
* Generic get_info
|
|
**************************************************************************/
|
|
|
|
DEVICE_GET_INFO( hc55516 )
|
|
{
|
|
switch (state)
|
|
{
|
|
/* --- the following bits of info are returned as 64-bit signed integers --- */
|
|
case DEVINFO_INT_TOKEN_BYTES: info->i = sizeof(hc55516_state); break;
|
|
|
|
/* --- the following bits of info are returned as pointers to data or functions --- */
|
|
case DEVINFO_FCT_START: info->start = DEVICE_START_NAME( hc55516 ); break;
|
|
case DEVINFO_FCT_RESET: info->reset = DEVICE_RESET_NAME( hc55516 ); break;
|
|
|
|
/* --- the following bits of info are returned as NULL-terminated strings --- */
|
|
case DEVINFO_STR_NAME: strcpy(info->s, "HC-55516"); break;
|
|
case DEVINFO_STR_FAMILY: strcpy(info->s, "CVSD"); break;
|
|
case DEVINFO_STR_VERSION: strcpy(info->s, "2.1"); break;
|
|
case DEVINFO_STR_SOURCE_FILE: strcpy(info->s, __FILE__); break;
|
|
case DEVINFO_STR_CREDITS: strcpy(info->s, "Copyright Nicola Salmoria and the MAME Team"); break;
|
|
}
|
|
}
|
|
|
|
|
|
DEVICE_GET_INFO( mc3417 )
|
|
{
|
|
switch (state)
|
|
{
|
|
case DEVINFO_FCT_START: info->start = DEVICE_START_NAME( mc3417 ); break;
|
|
case DEVINFO_FCT_RESET: /* chip has no reset pin */ break;
|
|
case DEVINFO_STR_NAME: strcpy(info->s, "MC3417"); break;
|
|
default: DEVICE_GET_INFO_CALL(hc55516); break;
|
|
}
|
|
}
|
|
|
|
|
|
DEVICE_GET_INFO( mc3418 )
|
|
{
|
|
switch (state)
|
|
{
|
|
case DEVINFO_FCT_START: info->start = DEVICE_START_NAME( mc3418 ); break;
|
|
case DEVINFO_FCT_RESET: /* chip has no reset pin */ break;
|
|
case DEVINFO_STR_NAME: strcpy(info->s, "MC3418"); break;
|
|
default: DEVICE_GET_INFO_CALL(hc55516); break;
|
|
}
|
|
}
|
|
|
|
|
|
DEFINE_LEGACY_SOUND_DEVICE(HC55516, hc55516);
|
|
DEFINE_LEGACY_SOUND_DEVICE(MC3417, mc3417);
|
|
DEFINE_LEGACY_SOUND_DEVICE(MC3418, mc3418);
|