2022-10-16 15:08:02 +00:00
|
|
|
package main
|
|
|
|
|
|
|
|
import (
|
|
|
|
"github.com/faiface/beep"
|
|
|
|
"github.com/faiface/beep/effects"
|
|
|
|
"github.com/faiface/beep/speaker"
|
|
|
|
"github.com/faiface/beep/wav"
|
|
|
|
"github.com/tarm/serial"
|
|
|
|
"log"
|
|
|
|
"math"
|
|
|
|
"os"
|
|
|
|
"sync"
|
|
|
|
"time"
|
2022-10-16 16:47:51 +00:00
|
|
|
"fmt"
|
2022-10-16 15:08:02 +00:00
|
|
|
)
|
|
|
|
|
|
|
|
var sounds = []*beep.Buffer{
|
2022-10-16 16:47:51 +00:00
|
|
|
mustLoadStream("./default/001.wav"),
|
|
|
|
mustLoadStream("./default/002.wav"),
|
|
|
|
mustLoadStream("./default/003.wav"),
|
|
|
|
mustLoadStream("./default/004.wav"),
|
|
|
|
mustLoadStream("./default/005.wav"),
|
|
|
|
mustLoadStream("./default/006.wav"),
|
|
|
|
mustLoadStream("./default/007.wav"),
|
|
|
|
mustLoadStream("./default/008.wav"),
|
2022-10-16 15:08:02 +00:00
|
|
|
}
|
|
|
|
|
2022-10-16 16:47:51 +00:00
|
|
|
var counter int
|
|
|
|
|
2022-10-16 15:08:02 +00:00
|
|
|
func mustLoadStream(p string) *beep.Buffer {
|
2022-10-16 16:47:51 +00:00
|
|
|
counter++
|
|
|
|
log.Println("Loading: /media/usb/00" + fmt.Sprintf("%d", counter) + ".wav")
|
|
|
|
f, err := os.Open("/media/usb/00" + fmt.Sprintf("%d", counter) + ".wav")
|
2022-10-16 15:08:02 +00:00
|
|
|
if err != nil {
|
2022-10-16 16:47:51 +00:00
|
|
|
log.Println("Failed: /media/usb/00" + fmt.Sprintf("%d", counter) + ".wav")
|
|
|
|
log.Println("Loading " + p)
|
|
|
|
f, err = os.Open(p)
|
|
|
|
if err != nil {
|
|
|
|
// log.Fatal(err)
|
|
|
|
return nil
|
|
|
|
}
|
2022-10-16 15:08:02 +00:00
|
|
|
}
|
|
|
|
defer f.Close()
|
|
|
|
|
|
|
|
streamer, format, err := wav.Decode(f)
|
|
|
|
if err != nil {
|
2022-10-16 16:47:51 +00:00
|
|
|
log.Fatal(err)
|
2022-10-16 15:08:02 +00:00
|
|
|
}
|
|
|
|
defer streamer.Close()
|
|
|
|
|
|
|
|
newFormat := format
|
2022-10-16 17:19:19 +00:00
|
|
|
newFormat.SampleRate = format.SampleRate / 4
|
2022-10-16 16:47:51 +00:00
|
|
|
// newFormat.SampleRate = format.SampleRate / 2
|
2022-10-16 17:19:19 +00:00
|
|
|
// newFormat.SampleRate = format.SampleRate
|
2022-10-16 15:08:02 +00:00
|
|
|
resample := beep.Resample(4, format.SampleRate, newFormat.SampleRate, streamer)
|
|
|
|
|
|
|
|
buffer := beep.NewBuffer(newFormat)
|
|
|
|
buffer.Append(resample)
|
|
|
|
|
|
|
|
return buffer
|
|
|
|
}
|
|
|
|
|
|
|
|
type soundWorker struct {
|
|
|
|
buffer *beep.Buffer
|
|
|
|
ctrl *beep.Ctrl
|
|
|
|
volume *effects.Volume
|
|
|
|
isRunning bool
|
|
|
|
isStopping bool
|
|
|
|
forceEnd bool
|
|
|
|
mtx sync.Mutex
|
|
|
|
}
|
|
|
|
|
|
|
|
func (w *soundWorker) End() {
|
|
|
|
w.isStopping = true
|
|
|
|
w.mtx.Lock()
|
|
|
|
defer w.mtx.Unlock()
|
|
|
|
|
|
|
|
i := 1
|
|
|
|
for w.isRunning {
|
|
|
|
if i >= 80 || w.forceEnd {
|
|
|
|
break
|
|
|
|
}
|
|
|
|
|
|
|
|
w.volume.Volume = float64(1) / math.Pow(10, float64(i))
|
|
|
|
i++
|
|
|
|
}
|
|
|
|
w.isRunning = false
|
|
|
|
w.isStopping = false
|
|
|
|
w.forceEnd = false
|
|
|
|
w.ctrl.Paused = true
|
|
|
|
}
|
|
|
|
|
|
|
|
func (w *soundWorker) Start() {
|
|
|
|
if w.isRunning {
|
|
|
|
w.forceEnd = true
|
|
|
|
}
|
|
|
|
|
|
|
|
w.mtx.Lock()
|
|
|
|
defer w.mtx.Unlock()
|
|
|
|
|
|
|
|
w.isRunning = true
|
2022-10-16 16:47:51 +00:00
|
|
|
// w.ctrl.Streamer.(beep.StreamSeeker).Seek(0)
|
|
|
|
// log.Println("Buffer Start: " + fmt.Sprintf("%d", w.ctrl.Streamer))
|
|
|
|
w.ctrl.Streamer = beep.Seq(w.buffer.Streamer(0, w.buffer.Len()), beep.Silence(-1))
|
|
|
|
// w.ctrl.Streamer = w.buffer.Streamer(0, w.buffer.Len())
|
2022-10-16 15:08:02 +00:00
|
|
|
w.volume.Volume = 1
|
|
|
|
w.ctrl.Paused = false
|
|
|
|
}
|
|
|
|
|
|
|
|
func main() {
|
|
|
|
config := &serial.Config{
|
|
|
|
Name: "/dev/ttyS0",
|
|
|
|
Baud: 115200,
|
|
|
|
Size: 8,
|
|
|
|
}
|
|
|
|
|
|
|
|
stream, err := serial.OpenPort(config)
|
|
|
|
if err != nil {
|
|
|
|
log.Fatal(err)
|
|
|
|
}
|
|
|
|
|
|
|
|
format := sounds[0].Format()
|
2022-10-16 16:47:51 +00:00
|
|
|
// if err := speaker.Init(format.SampleRate, format.SampleRate.N(time.Second/100)); err != nil {
|
2022-10-16 15:08:02 +00:00
|
|
|
if err := speaker.Init(format.SampleRate, format.SampleRate.N(time.Second/100)); err != nil {
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
var soundWorkers []*soundWorker
|
|
|
|
|
|
|
|
mixer := beep.Mixer{}
|
|
|
|
for _, buffer := range sounds {
|
2022-10-16 16:47:51 +00:00
|
|
|
if buffer == nil {
|
|
|
|
continue
|
|
|
|
}
|
2022-10-16 15:08:02 +00:00
|
|
|
c := &beep.Ctrl{
|
2022-10-16 16:47:51 +00:00
|
|
|
// Streamer: buffer.Streamer(0, buffer.Len()),
|
|
|
|
Streamer: beep.Seq(buffer.Streamer(0, buffer.Len()), beep.Silence(-1)),
|
2022-10-16 15:08:02 +00:00
|
|
|
Paused: true,
|
|
|
|
}
|
|
|
|
|
|
|
|
volume := &effects.Volume{
|
|
|
|
Streamer: c,
|
2022-10-16 16:47:51 +00:00
|
|
|
// Streamer: beep.Seq(c, beep.Silence(-1)),
|
2022-10-16 15:08:02 +00:00
|
|
|
Base: 2,
|
|
|
|
Volume: 1,
|
|
|
|
}
|
|
|
|
|
|
|
|
mixer.Add(volume)
|
|
|
|
|
|
|
|
soundWorkers = append(soundWorkers, &soundWorker{
|
|
|
|
buffer: buffer,
|
|
|
|
ctrl: c,
|
|
|
|
volume: volume,
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
|
|
|
speaker.Play(&mixer)
|
|
|
|
|
|
|
|
c := make(chan byte)
|
|
|
|
|
2022-10-16 16:47:51 +00:00
|
|
|
// List das byte von serial ein und prueft die bits
|
2022-10-16 15:08:02 +00:00
|
|
|
go func() {
|
|
|
|
for b := range c {
|
|
|
|
for i, w := range soundWorkers {
|
|
|
|
setStop := (b&(1<<i))>>i == 0
|
|
|
|
|
|
|
|
if w.isRunning && !w.isStopping && setStop {
|
|
|
|
w.End()
|
|
|
|
}
|
|
|
|
|
|
|
|
if !w.isRunning && !setStop {
|
|
|
|
w.Start()
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}()
|
|
|
|
|
2022-10-16 16:47:51 +00:00
|
|
|
// List das byte ein
|
2022-10-16 15:08:02 +00:00
|
|
|
for {
|
|
|
|
buf := make([]byte, 1)
|
|
|
|
_, err := stream.Read(buf)
|
|
|
|
if err != nil {
|
|
|
|
log.Fatal(err)
|
|
|
|
}
|
|
|
|
|
|
|
|
c <- buf[0]
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|