pnyx/cmd/client/main.go

263 lines
6.3 KiB
Go

2024-04-03 18:52:04 -06:00
package main
import (
2024-04-08 17:23:55 -06:00
"encoding/binary"
"fmt"
"os"
"time"
"git.metznet.ca/MetzNet/pnyx"
2024-04-08 17:23:55 -06:00
"github.com/gen2brain/malgo"
"github.com/hraban/opus"
2024-04-03 18:52:04 -06:00
)
func main() {
decoders := map[pnyx.PeerID]chan[]byte{}
2024-04-08 17:26:43 -06:00
encoder, err := opus.NewEncoder(48000, 1, opus.AppVoIP)
2024-04-08 17:23:55 -06:00
if err != nil {
panic(err)
}
ctx, err := malgo.InitContext(nil, malgo.ContextConfig{}, nil)
if err != nil {
panic(err)
}
defer ctx.Free()
defer ctx.Uninit()
2024-04-08 11:32:39 -06:00
infos, err := ctx.Devices(malgo.Playback)
if err != nil {
panic(err)
}
2024-04-08 11:56:34 -06:00
var playback_device *malgo.DeviceInfo = nil
2024-04-08 11:46:43 -06:00
for _, info := range infos {
if info.IsDefault != 0 {
2024-04-08 11:56:34 -06:00
full, err := ctx.DeviceInfo(malgo.Playback, info.ID, malgo.Shared)
if err != nil {
panic(err)
}
playback_device = &full
2024-04-08 11:46:43 -06:00
}
2024-04-08 11:32:39 -06:00
}
2024-04-08 11:56:34 -06:00
if playback_device == nil {
panic("No default playback device")
}
2024-04-08 11:32:39 -06:00
infos, err = ctx.Devices(malgo.Capture)
if err != nil {
panic(err)
}
2024-04-08 11:56:34 -06:00
var capture_device *malgo.DeviceInfo = nil
2024-04-08 11:46:43 -06:00
for _, info := range infos {
if info.IsDefault != 0 {
2024-04-08 11:56:34 -06:00
full, err := ctx.DeviceInfo(malgo.Capture, info.ID, malgo.Shared)
if err != nil {
panic(err)
}
capture_device = &full
2024-04-08 11:46:43 -06:00
}
2024-04-08 11:32:39 -06:00
}
2024-04-08 11:56:34 -06:00
if capture_device == nil {
panic("No default capture device")
}
inDeviceConfig := malgo.DefaultDeviceConfig(malgo.Capture)
2024-04-08 17:23:55 -06:00
inDeviceConfig.Capture.Format = malgo.FormatS16
2024-04-08 12:08:36 -06:00
inDeviceConfig.Capture.Channels = 1
2024-04-08 11:47:56 -06:00
inDeviceConfig.Capture.DeviceID = capture_device.ID.Pointer()
2024-04-08 17:26:43 -06:00
inDeviceConfig.SampleRate = 48000
inDeviceConfig.PeriodSizeInFrames = 960
inDeviceConfig.Alsa.NoMMap = 1
inDeviceConfig.Capture.ShareMode = malgo.Shared
outDeviceConfig := malgo.DefaultDeviceConfig(malgo.Playback)
2024-04-08 17:23:55 -06:00
outDeviceConfig.Playback.Format = malgo.FormatS16
2024-04-08 12:08:36 -06:00
outDeviceConfig.Playback.Channels = 1
2024-04-08 11:47:56 -06:00
outDeviceConfig.Playback.DeviceID = playback_device.ID.Pointer()
2024-04-08 17:26:43 -06:00
outDeviceConfig.SampleRate = 48000
outDeviceConfig.PeriodSizeInFrames = 960
outDeviceConfig.Alsa.NoMMap = 1
outDeviceConfig.Playback.ShareMode = malgo.Shared
mic := make(chan []byte, 0)
speaker := make(chan []byte, 0)
onSendFrames := func(output_samples []byte, input_samples []byte, framecount uint32) {
select {
case data := <- speaker:
copy(output_samples, data)
2024-04-08 11:59:36 -06:00
default:
}
}
playbackCallbacks := malgo.DeviceCallbacks{
Data: onSendFrames,
}
2024-04-08 11:58:32 -06:00
fmt.Printf("Creating playback device %+v with format %+v\n", playback_device, playback_device.Formats[0])
outDevice, err := malgo.InitDevice(ctx.Context, outDeviceConfig, playbackCallbacks)
if err != nil {
panic(err)
}
2024-04-08 11:46:43 -06:00
fmt.Printf("Starting playback\n")
err = outDevice.Start()
if err != nil {
panic(err)
}
defer outDevice.Uninit()
defer outDevice.Stop()
onRecvFrames := func(output_samples []byte, input_samples []byte, framecount uint32) {
2024-04-08 17:23:55 -06:00
pcm := make([]int16, len(input_samples)/2)
for i := 0; i < len(input_samples)/2; i++ {
2024-04-08 17:32:49 -06:00
pcm[i] = int16(binary.LittleEndian.Uint16(input_samples[2*i:]))
2024-04-08 17:23:55 -06:00
}
data := make([]byte, len(input_samples))
2024-04-08 17:23:55 -06:00
written, err := encoder.Encode(pcm, data)
if err != nil {
panic(err)
}
select {
2024-04-08 17:23:55 -06:00
case mic <- data[:written]:
default:
}
}
captureCallbacks := malgo.DeviceCallbacks{
Data: onRecvFrames,
}
2024-04-08 11:58:32 -06:00
fmt.Printf("Creating capture device %+v with format %+v\n", capture_device, capture_device.Formats[0])
inDevice, err := malgo.InitDevice(ctx.Context, inDeviceConfig, captureCallbacks)
if err != nil {
panic(err)
}
2024-04-08 11:35:38 -06:00
fmt.Printf("Starting capture device\n")
err = inDevice.Start()
if err != nil {
panic(err)
}
defer inDevice.Uninit()
defer inDevice.Stop()
fmt.Printf("Starting pnyx client\n")
2024-04-06 17:03:31 -06:00
client, err := pnyx.NewClient(nil, os.Args[1])
2024-04-03 18:52:04 -06:00
if err != nil {
panic(err)
}
go func() {
var buf [1024]byte
for true {
read, _, err := client.Connection.ReadFromUDP(buf[:])
if err != nil {
fmt.Printf("Read Error %s\n", err)
break
}
data, err := pnyx.ParseSessionData(&client.Session, buf[pnyx.COMMAND_LENGTH + pnyx.ID_LENGTH:read])
if err != nil {
fmt.Printf("ParseSessionData Error %s\n", err)
continue
}
packet, err := pnyx.ParsePacket(data)
if err != nil {
fmt.Printf("ParsePacket Error %s - %x\n", err, data)
continue
}
2024-04-08 17:23:55 -06:00
peer := pnyx.PeerID(packet.Data[0:16])
if packet.Channel == pnyx.ChannelID(1) {
decode_chan, exists := decoders[peer]
2024-04-08 17:23:55 -06:00
if exists == false {
decode_chan = make(chan[]byte, 1000)
decoders[peer] = decode_chan
go func(decode_chan chan[]byte){
decoder, err := opus.NewDecoder(48000, 1)
if err != nil {
panic(err)
}
for true {
select {
case <-time.After(20*time.Millisecond):
pcm := make([]int16, 960)
err := decoder.DecodePLC(pcm)
if err != nil {
panic(err)
}
pcm_bytes := make([]byte, 960*2)
for i := 0; i < 960; i++ {
binary.LittleEndian.PutUint16(pcm_bytes[i*2:], uint16(pcm[i]))
}
speaker <- pcm_bytes
case data := <-decode_chan:
pcm := make([]int16, 960)
written, err := decoder.Decode(data, pcm)
if err != nil {
panic(err)
}
pcm_bytes := make([]byte, written*2)
for i := 0; i < written; i++ {
binary.LittleEndian.PutUint16(pcm_bytes[i*2:], uint16(pcm[i]))
}
speaker <- pcm_bytes
}
}
}(decoders[peer])
2024-04-08 17:23:55 -06:00
}
decode_chan <- packet.Data[16:]
2024-04-08 17:23:55 -06:00
}
}
}()
err = client.Send(pnyx.Packet{
Channel: pnyx.ChannelID(1),
Mode: pnyx.MODE_RAW,
Command: pnyx.MODE_COMMAND_JOIN,
Data: nil,
})
if err != nil {
panic(err)
}
2024-04-08 17:23:55 -06:00
err = client.Send(pnyx.Packet{
Channel: pnyx.ChannelID(2),
Mode: pnyx.MODE_RAW,
Command: pnyx.MODE_COMMAND_JOIN,
Data: nil,
})
if err != nil {
panic(err)
}
for true {
data := <- mic
err = client.Send(pnyx.Packet{
Channel: pnyx.ChannelID(1),
Mode: pnyx.MODE_RAW,
Command: pnyx.MODE_COMMAND_DATA,
Data: data,
})
if err != nil {
panic(err)
}
}
2024-04-03 18:52:04 -06:00
}