annotate src/audio.c @ 59:b15187a99d6f default tip

Add a command line option for printing out label addresses on the command line. Useful for debugging purposes.
author Michael Pavone <pavone@retrodev.com>
date Wed, 07 Sep 2016 23:15:27 -0700
parents 23bea9b9569f
children
Ignore whitespace changes - Everywhere: Within whitespace: At end of lines:
rev   line source
24
4c9dbfa30a66 Implemented audio
Michael Pavone <pavone@retrodev.com>
parents:
diff changeset
1 #include <stdint.h>
4c9dbfa30a66 Implemented audio
Michael Pavone <pavone@retrodev.com>
parents:
diff changeset
2 #include <stdlib.h>
4c9dbfa30a66 Implemented audio
Michael Pavone <pavone@retrodev.com>
parents:
diff changeset
3 #include <string.h>
4c9dbfa30a66 Implemented audio
Michael Pavone <pavone@retrodev.com>
parents:
diff changeset
4 #include "audio.h"
4c9dbfa30a66 Implemented audio
Michael Pavone <pavone@retrodev.com>
parents:
diff changeset
5 #include "system.h"
4c9dbfa30a66 Implemented audio
Michael Pavone <pavone@retrodev.com>
parents:
diff changeset
6
4c9dbfa30a66 Implemented audio
Michael Pavone <pavone@retrodev.com>
parents:
diff changeset
7 #define BUFFER_INC_RES 1000000000UL
4c9dbfa30a66 Implemented audio
Michael Pavone <pavone@retrodev.com>
parents:
diff changeset
8
4c9dbfa30a66 Implemented audio
Michael Pavone <pavone@retrodev.com>
parents:
diff changeset
9 audio *alloc_audio(uint32_t master_clock, uint32_t clock_div, int sample_rate, int buffer_size)
4c9dbfa30a66 Implemented audio
Michael Pavone <pavone@retrodev.com>
parents:
diff changeset
10 {
4c9dbfa30a66 Implemented audio
Michael Pavone <pavone@retrodev.com>
parents:
diff changeset
11 size_t alloc_size = sizeof(audio) + buffer_size * sizeof(int16_t) * 2;
4c9dbfa30a66 Implemented audio
Michael Pavone <pavone@retrodev.com>
parents:
diff changeset
12 audio *context = malloc(alloc_size);
4c9dbfa30a66 Implemented audio
Michael Pavone <pavone@retrodev.com>
parents:
diff changeset
13 memset(context, 0, alloc_size);
4c9dbfa30a66 Implemented audio
Michael Pavone <pavone@retrodev.com>
parents:
diff changeset
14 context->writebuffer = context->buffer;
4c9dbfa30a66 Implemented audio
Michael Pavone <pavone@retrodev.com>
parents:
diff changeset
15 context->playbuffer = context->buffer + buffer_size;
4c9dbfa30a66 Implemented audio
Michael Pavone <pavone@retrodev.com>
parents:
diff changeset
16
4c9dbfa30a66 Implemented audio
Michael Pavone <pavone@retrodev.com>
parents:
diff changeset
17 context->buffer_size = buffer_size;
4c9dbfa30a66 Implemented audio
Michael Pavone <pavone@retrodev.com>
parents:
diff changeset
18 context->clock_inc = clock_div;
4c9dbfa30a66 Implemented audio
Michael Pavone <pavone@retrodev.com>
parents:
diff changeset
19 context->buffer_inc = ((BUFFER_INC_RES * (uint64_t)sample_rate) / (uint64_t)master_clock) * clock_div;
4c9dbfa30a66 Implemented audio
Michael Pavone <pavone@retrodev.com>
parents:
diff changeset
20 for (int i = 0; i < NUM_CHANNELS; i++)
4c9dbfa30a66 Implemented audio
Michael Pavone <pavone@retrodev.com>
parents:
diff changeset
21 {
4c9dbfa30a66 Implemented audio
Michael Pavone <pavone@retrodev.com>
parents:
diff changeset
22 context->value[i] = context->volume[i] << 5;
4c9dbfa30a66 Implemented audio
Michael Pavone <pavone@retrodev.com>
parents:
diff changeset
23 }
41
23bea9b9569f Fix silly bug in alloc_audio
Michael Pavone <pavone@retrodev.com>
parents: 24
diff changeset
24 return context;
24
4c9dbfa30a66 Implemented audio
Michael Pavone <pavone@retrodev.com>
parents:
diff changeset
25 }
4c9dbfa30a66 Implemented audio
Michael Pavone <pavone@retrodev.com>
parents:
diff changeset
26
4c9dbfa30a66 Implemented audio
Michael Pavone <pavone@retrodev.com>
parents:
diff changeset
27 void audio_run(audio *context, uint32_t target)
4c9dbfa30a66 Implemented audio
Michael Pavone <pavone@retrodev.com>
parents:
diff changeset
28 {
4c9dbfa30a66 Implemented audio
Michael Pavone <pavone@retrodev.com>
parents:
diff changeset
29 while (context->cycles < target)
4c9dbfa30a66 Implemented audio
Michael Pavone <pavone@retrodev.com>
parents:
diff changeset
30 {
4c9dbfa30a66 Implemented audio
Michael Pavone <pavone@retrodev.com>
parents:
diff changeset
31 for (int i = 0; i < 4; i++)
4c9dbfa30a66 Implemented audio
Michael Pavone <pavone@retrodev.com>
parents:
diff changeset
32 {
4c9dbfa30a66 Implemented audio
Michael Pavone <pavone@retrodev.com>
parents:
diff changeset
33 if (context->timer_cur[i]) {
4c9dbfa30a66 Implemented audio
Michael Pavone <pavone@retrodev.com>
parents:
diff changeset
34 context->timer_cur[i]--;
4c9dbfa30a66 Implemented audio
Michael Pavone <pavone@retrodev.com>
parents:
diff changeset
35 if (!context->timer_cur[i]) {
4c9dbfa30a66 Implemented audio
Michael Pavone <pavone@retrodev.com>
parents:
diff changeset
36 context->value[i] = context->value[i] ? 0 : (context->volume[i] << 5);
4c9dbfa30a66 Implemented audio
Michael Pavone <pavone@retrodev.com>
parents:
diff changeset
37 }
4c9dbfa30a66 Implemented audio
Michael Pavone <pavone@retrodev.com>
parents:
diff changeset
38 } else {
4c9dbfa30a66 Implemented audio
Michael Pavone <pavone@retrodev.com>
parents:
diff changeset
39 context->timer_cur[i] = context->timer_load[i];
4c9dbfa30a66 Implemented audio
Michael Pavone <pavone@retrodev.com>
parents:
diff changeset
40 }
4c9dbfa30a66 Implemented audio
Michael Pavone <pavone@retrodev.com>
parents:
diff changeset
41 }
4c9dbfa30a66 Implemented audio
Michael Pavone <pavone@retrodev.com>
parents:
diff changeset
42 context->buffer_fraction += context->buffer_inc;
4c9dbfa30a66 Implemented audio
Michael Pavone <pavone@retrodev.com>
parents:
diff changeset
43 if (context->buffer_fraction >= BUFFER_INC_RES) {
4c9dbfa30a66 Implemented audio
Michael Pavone <pavone@retrodev.com>
parents:
diff changeset
44 context->buffer_fraction -= BUFFER_INC_RES;
4c9dbfa30a66 Implemented audio
Michael Pavone <pavone@retrodev.com>
parents:
diff changeset
45 context->writebuffer[context->buffer_pos++] =
4c9dbfa30a66 Implemented audio
Michael Pavone <pavone@retrodev.com>
parents:
diff changeset
46 context->value[0] + context->value[1] + context->value[2] + context->value[3];
4c9dbfa30a66 Implemented audio
Michael Pavone <pavone@retrodev.com>
parents:
diff changeset
47 if (context->buffer_pos == context->buffer_size) {
4c9dbfa30a66 Implemented audio
Michael Pavone <pavone@retrodev.com>
parents:
diff changeset
48 int16_t *tmp = context->playbuffer;
4c9dbfa30a66 Implemented audio
Michael Pavone <pavone@retrodev.com>
parents:
diff changeset
49 context->playbuffer = context->writebuffer;
4c9dbfa30a66 Implemented audio
Michael Pavone <pavone@retrodev.com>
parents:
diff changeset
50 context->writebuffer = tmp;
4c9dbfa30a66 Implemented audio
Michael Pavone <pavone@retrodev.com>
parents:
diff changeset
51 system_present_audio(context->playbuffer);
4c9dbfa30a66 Implemented audio
Michael Pavone <pavone@retrodev.com>
parents:
diff changeset
52 context->buffer_pos = 0;
4c9dbfa30a66 Implemented audio
Michael Pavone <pavone@retrodev.com>
parents:
diff changeset
53 }
4c9dbfa30a66 Implemented audio
Michael Pavone <pavone@retrodev.com>
parents:
diff changeset
54 }
4c9dbfa30a66 Implemented audio
Michael Pavone <pavone@retrodev.com>
parents:
diff changeset
55 context->cycles += context->clock_inc;
4c9dbfa30a66 Implemented audio
Michael Pavone <pavone@retrodev.com>
parents:
diff changeset
56 }
4c9dbfa30a66 Implemented audio
Michael Pavone <pavone@retrodev.com>
parents:
diff changeset
57 }
4c9dbfa30a66 Implemented audio
Michael Pavone <pavone@retrodev.com>
parents:
diff changeset
58
4c9dbfa30a66 Implemented audio
Michael Pavone <pavone@retrodev.com>
parents:
diff changeset
59 void audio_write_freq(audio *context, int channel, uint16_t value)
4c9dbfa30a66 Implemented audio
Michael Pavone <pavone@retrodev.com>
parents:
diff changeset
60 {
4c9dbfa30a66 Implemented audio
Michael Pavone <pavone@retrodev.com>
parents:
diff changeset
61 context->timer_cur[channel] = context->timer_load[channel] = value;
4c9dbfa30a66 Implemented audio
Michael Pavone <pavone@retrodev.com>
parents:
diff changeset
62 }
4c9dbfa30a66 Implemented audio
Michael Pavone <pavone@retrodev.com>
parents:
diff changeset
63
4c9dbfa30a66 Implemented audio
Michael Pavone <pavone@retrodev.com>
parents:
diff changeset
64 void audio_write_vol(audio *context, int pair, uint16_t value)
4c9dbfa30a66 Implemented audio
Michael Pavone <pavone@retrodev.com>
parents:
diff changeset
65 {
4c9dbfa30a66 Implemented audio
Michael Pavone <pavone@retrodev.com>
parents:
diff changeset
66 int channel = pair * 2;
4c9dbfa30a66 Implemented audio
Michael Pavone <pavone@retrodev.com>
parents:
diff changeset
67 context->value[channel] = context->volume[channel] = value >> 8;
4c9dbfa30a66 Implemented audio
Michael Pavone <pavone@retrodev.com>
parents:
diff changeset
68 channel++;
4c9dbfa30a66 Implemented audio
Michael Pavone <pavone@retrodev.com>
parents:
diff changeset
69 context->value[channel] = context->volume[channel] = value;
4c9dbfa30a66 Implemented audio
Michael Pavone <pavone@retrodev.com>
parents:
diff changeset
70 }