2019-06-01 15:11:25 +00:00
|
|
|
package main
|
2019-05-09 22:09:15 +00:00
|
|
|
|
|
|
|
/*
|
|
|
|
typedef unsigned char Uint8;
|
|
|
|
void SpeakerCallback(void *userdata, Uint8 *stream, int len);
|
|
|
|
*/
|
|
|
|
import "C"
|
|
|
|
import (
|
2019-05-10 16:07:36 +00:00
|
|
|
"fmt"
|
2019-05-09 22:09:15 +00:00
|
|
|
"reflect"
|
|
|
|
"unsafe"
|
|
|
|
|
2019-06-01 15:11:25 +00:00
|
|
|
"github.com/ivanizag/apple2"
|
2019-05-09 22:09:15 +00:00
|
|
|
"github.com/veandco/go-sdl2/sdl"
|
|
|
|
)
|
|
|
|
|
|
|
|
const (
|
|
|
|
samplingHz = 48000
|
2019-11-20 22:57:24 +00:00
|
|
|
bufferSize = 1000
|
2019-05-09 22:09:15 +00:00
|
|
|
// bufferSize/samplingHz will be the max delay of the sound
|
2019-11-07 22:20:14 +00:00
|
|
|
sampleDurationCycles = 1000000 * apple2.CPUClockMhz / samplingHz
|
2019-05-09 22:09:15 +00:00
|
|
|
// each sample on the sound stream is 21.31 cpu cycles approx
|
|
|
|
maxOutOfSyncMs = 2000
|
2019-10-19 18:33:50 +00:00
|
|
|
decayLevel = 128
|
2019-05-09 22:09:15 +00:00
|
|
|
)
|
|
|
|
|
|
|
|
type sdlSpeaker struct {
|
|
|
|
clickChannel chan uint64
|
|
|
|
pendingClicks []uint64
|
|
|
|
lastCycle uint64
|
|
|
|
lastState bool
|
2019-10-19 18:33:50 +00:00
|
|
|
lastLevel C.Uint8
|
2019-05-09 22:09:15 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
I have not found a way to encode the pointer to sdlSpeaker on the userdata of
|
|
|
|
the call to SpeakerCallback(). I use a global as workaround...
|
|
|
|
*/
|
2019-06-01 15:11:25 +00:00
|
|
|
var theSDLSpeaker *sdlSpeaker
|
2019-05-09 22:09:15 +00:00
|
|
|
|
2019-06-01 15:11:25 +00:00
|
|
|
func newSDLSpeaker() *sdlSpeaker {
|
2019-05-09 22:09:15 +00:00
|
|
|
var s sdlSpeaker
|
|
|
|
s.clickChannel = make(chan uint64, bufferSize)
|
|
|
|
s.pendingClicks = make([]uint64, 0, bufferSize)
|
2019-10-19 18:33:50 +00:00
|
|
|
s.lastLevel = decayLevel // Mid position to avoid starting clicks.
|
2019-05-09 22:09:15 +00:00
|
|
|
return &s
|
|
|
|
}
|
|
|
|
|
|
|
|
// Click receives a speaker click. The argument is the CPU cycle when it is generated
|
|
|
|
func (s *sdlSpeaker) Click(cycle uint64) {
|
|
|
|
s.clickChannel <- cycle
|
|
|
|
}
|
|
|
|
|
|
|
|
func stateToLevel(state bool) C.Uint8 {
|
|
|
|
if state {
|
|
|
|
return 255
|
|
|
|
}
|
|
|
|
return 0
|
|
|
|
}
|
|
|
|
|
2019-10-19 19:01:09 +00:00
|
|
|
// SpeakerCallback is called to get more sound buffer data
|
2019-05-09 22:09:15 +00:00
|
|
|
//export SpeakerCallback
|
|
|
|
func SpeakerCallback(userdata unsafe.Pointer, stream *C.Uint8, length C.int) {
|
2019-06-01 15:11:25 +00:00
|
|
|
s := theSDLSpeaker
|
2019-05-09 22:09:15 +00:00
|
|
|
if s == nil {
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
// Adapt C buffer
|
|
|
|
n := int(length)
|
|
|
|
hdr := reflect.SliceHeader{Data: uintptr(unsafe.Pointer(stream)), Len: n, Cap: n}
|
|
|
|
buf := *(*[]C.Uint8)(unsafe.Pointer(&hdr))
|
|
|
|
|
|
|
|
//Read queued clicks
|
|
|
|
done := false
|
|
|
|
for !done {
|
|
|
|
select {
|
|
|
|
case cycle := <-s.clickChannel:
|
|
|
|
s.pendingClicks = append(s.pendingClicks, cycle)
|
|
|
|
default:
|
|
|
|
done = true
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Verify that we are not too long behind
|
2019-11-07 22:20:14 +00:00
|
|
|
var maxOutOfSyncCyclesFloat = 1000 * apple2.CPUClockMhz * maxOutOfSyncMs
|
2019-05-09 22:09:15 +00:00
|
|
|
var maxOutOfSyncCycles = uint64(maxOutOfSyncCyclesFloat)
|
|
|
|
for _, pc := range s.pendingClicks {
|
|
|
|
if pc-s.lastCycle > maxOutOfSyncCycles {
|
|
|
|
// Fast forward
|
|
|
|
s.lastCycle = pc
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Build wave
|
|
|
|
var i, p int
|
2019-10-19 18:33:50 +00:00
|
|
|
level := s.lastLevel
|
2019-05-09 22:09:15 +00:00
|
|
|
for p = 0; p < len(s.pendingClicks); p++ {
|
|
|
|
cycle := s.pendingClicks[p]
|
|
|
|
if cycle < s.lastCycle {
|
|
|
|
// Too old, ignore
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
|
|
|
|
// Fill with samples
|
|
|
|
samplesNeeded := int(float64(cycle-s.lastCycle) / sampleDurationCycles)
|
|
|
|
if samplesNeeded+i > bufferSize {
|
|
|
|
samplesNeeded = bufferSize - i
|
|
|
|
}
|
|
|
|
for j := 0; j < samplesNeeded; j++ {
|
|
|
|
buf[i] = level
|
|
|
|
i++
|
|
|
|
}
|
|
|
|
|
|
|
|
// Update state
|
|
|
|
s.lastCycle = cycle
|
|
|
|
s.lastState = !s.lastState
|
|
|
|
level = stateToLevel(s.lastState)
|
|
|
|
|
|
|
|
if i == bufferSize {
|
|
|
|
// Buffer is complete
|
|
|
|
break
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-10-19 18:33:50 +00:00
|
|
|
// If the buffer is empty lets decay the signal
|
|
|
|
if i == 0 {
|
|
|
|
for level != decayLevel && i < bufferSize {
|
|
|
|
if i%100 == 0 {
|
|
|
|
if level > decayLevel {
|
|
|
|
level--
|
|
|
|
} else {
|
|
|
|
level++
|
|
|
|
}
|
|
|
|
}
|
|
|
|
buf[i] = level
|
|
|
|
i++
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-05-09 22:09:15 +00:00
|
|
|
// Complete the buffer if needed
|
|
|
|
for b := i; b < bufferSize; b++ {
|
|
|
|
buf[b] = level
|
|
|
|
}
|
2019-10-19 18:33:50 +00:00
|
|
|
s.lastLevel = level
|
2019-05-09 22:09:15 +00:00
|
|
|
|
|
|
|
// Remove processed clicks, store the rest for later
|
|
|
|
remainingClicks := len(s.pendingClicks) - p
|
|
|
|
for r := 0; r < remainingClicks; r++ {
|
|
|
|
s.pendingClicks[r] = s.pendingClicks[p+r]
|
|
|
|
}
|
|
|
|
s.pendingClicks = s.pendingClicks[0:remainingClicks]
|
|
|
|
}
|
|
|
|
|
|
|
|
func (s *sdlSpeaker) start() {
|
|
|
|
err := sdl.Init(sdl.INIT_AUDIO)
|
|
|
|
if err != nil {
|
2019-05-10 16:07:36 +00:00
|
|
|
fmt.Printf("Error starting SDL audio: %v.\n", err)
|
2019-05-09 22:09:15 +00:00
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
spec := &sdl.AudioSpec{
|
|
|
|
Freq: samplingHz,
|
|
|
|
Format: sdl.AUDIO_U8,
|
|
|
|
Channels: 1,
|
|
|
|
Samples: bufferSize,
|
|
|
|
Callback: sdl.AudioCallback(C.SpeakerCallback),
|
|
|
|
}
|
|
|
|
|
|
|
|
if err := sdl.OpenAudio(spec, nil); err != nil {
|
2019-05-10 16:07:36 +00:00
|
|
|
fmt.Printf("Error opening the SDL audio channel: %v.\n", err)
|
2019-05-09 22:09:15 +00:00
|
|
|
return
|
|
|
|
}
|
|
|
|
sdl.PauseAudio(false)
|
2019-06-01 15:11:25 +00:00
|
|
|
theSDLSpeaker = s
|
2019-05-09 22:09:15 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
func (s *sdlSpeaker) close() {
|
|
|
|
sdl.CloseAudio()
|
|
|
|
sdl.Quit()
|
|
|
|
}
|