Browse Source

Make a PulseAudio audio file player.

pull/2/head
Justine Alexandra Roberts Tunney 11 years ago
parent
commit
16026e45c5
2 changed files with 180 additions and 0 deletions
  1. +1
    -0
      .gitignore
  2. +179
    -0
      fone/main.go

+ 1
- 0
.gitignore View File

@ -1 +1,2 @@
*.test
/fone/fone

+ 179
- 0
fone/main.go View File

@ -0,0 +1,179 @@
package main
// #cgo pkg-config: libpulse-simple
// #include <stdlib.h>
// #include <pulse/simple.h>
// #include <pulse/error.h>
import "C"
import (
"errors"
"flag"
"github.com/jart/gosip/dsp"
"github.com/jart/gosip/rtp"
"github.com/jart/gosip/sdp"
"github.com/jart/gosip/sip"
"github.com/jart/gosip/util"
"net"
"os"
"time"
"unsafe"
)
const (
hz = 8000
chans = 1
ptime = 20
ssize = 2
psamps = hz / (1000 / ptime) * chans
pbytes = psamps * ssize
filename = "/var/lib/asterisk/sounds/en/cc-yougotpranked.s16"
)
var (
address = flag.String("sipAddress", ":9020", "Listen address")
paServerFlag = flag.String("paServer", "", "Pulse Audio server name")
paSinkFlag = flag.String("paSink", "", "Pulse Audio device or sink name")
paName = C.CString("fone")
)
func main() {
pa, err := makePulseAudio(C.PA_STREAM_PLAYBACK, filename)
if err != nil {
panic(err)
}
defer C.pa_simple_free(pa)
defer C.pa_simple_flush(pa, nil)
f, err := os.Open(filename)
if err != nil {
panic(err)
}
defer f.Close()
tick := time.NewTicker(ptime * time.Millisecond)
defer tick.Stop()
func() {
for {
var buf [pbytes]byte
select {
case <-tick.C:
got, _ := f.Read(buf[:])
if got < pbytes {
return
}
var paerr C.int
if C.pa_simple_write(pa, unsafe.Pointer(&buf[0]), pbytes, &paerr) != 0 {
panic(C.GoString(C.pa_strerror(paerr)))
}
}
}
}()
os.Exit(0)
// Create RTP audio session.
rs, err := rtp.NewSession("")
if err != nil {
panic(err)
}
defer rs.Close()
rtpPort := uint16(rs.Sock.LocalAddr().(*net.UDPAddr).Port)
invite := &sip.Msg{
Method: sip.MethodInvite,
Request: &sip.URI{User: "echo", Host: "127.0.0.1", Port: 5060},
Payload: &sdp.SDP{
Origin: sdp.Origin{ID: util.GenerateOriginID()},
Audio: &sdp.Media{
Port: rtpPort,
Codecs: []sdp.Codec{sdp.ULAWCodec, sdp.DTMFCodec},
},
},
}
// Create a SIP phone call.
dl, err := sip.NewDialog(invite)
if err != nil {
panic(err)
}
// We're going to send white noise every 20ms.
var frame rtp.Frame
awgn := dsp.NewAWGN(-45.0)
ticker := time.NewTicker(20 * time.Millisecond)
defer ticker.Stop()
// Hangup after 200ms.
death := time.After(200 * time.Millisecond)
// Let's GO!
var answered bool
for {
select {
case <-ticker.C:
for n := 0; n < 160; n++ {
frame[n] = awgn.Get()
}
if err := rs.Send(&frame); err != nil {
panic("RTP send failed: " + err.Error())
}
case err := <-dl.OnErr:
panic(err)
case state := <-dl.OnState:
switch state {
case sip.DialogAnswered:
answered = true
case sip.DialogHangup:
if !answered {
panic("Call didn't get answered!")
}
return
}
case rs.Peer = <-dl.OnPeer:
case frame := <-rs.C:
rs.R <- frame
case err := <-rs.E:
panic("RTP recv failed: " + err.Error())
rs.CloseAfterError()
dl.Hangup <- true
case <-death:
dl.Hangup <- true
}
}
}
func makePulseAudio(direction C.pa_stream_direction_t, streamName string) (*C.pa_simple, error) {
var ss C.pa_sample_spec
ss.format = C.PA_SAMPLE_S16NE
ss.rate = hz
ss.channels = chans
var ba C.pa_buffer_attr
ba.maxlength = pbytes * 4
ba.tlength = pbytes
ba.prebuf = pbytes * 2
ba.minreq = pbytes
ba.fragsize = 0xffffffff
var paServer *C.char
if *paServerFlag != "" {
paServer = C.CString(*paServerFlag)
defer C.free(unsafe.Pointer(paServer))
}
var paSink *C.char
if *paSinkFlag != "" {
paSink = C.CString(*paSinkFlag)
defer C.free(unsafe.Pointer(paSink))
}
paStreamName := C.CString(streamName)
defer C.free(unsafe.Pointer(paStreamName))
var paerr C.int
pa := C.pa_simple_new(paServer, paName, direction, paSink, paStreamName, &ss, nil, &ba, &paerr)
if pa == nil {
return nil, errors.New(C.GoString(C.pa_strerror(paerr)))
}
return pa, nil
}

Loading…
Cancel
Save