mirror of
https://github.com/go-gst/go-gst.git
synced 2025-10-09 18:00:24 +08:00
85 lines
1.9 KiB
Go
85 lines
1.9 KiB
Go
package gst
|
|
|
|
// #include "gst.go.h"
|
|
import "C"
|
|
|
|
import (
|
|
"sync"
|
|
"time"
|
|
)
|
|
|
|
// Bus is a Go wrapper around a GstBus. It provides convenience methods for
|
|
// popping messages from the queue.
|
|
type Bus struct {
|
|
*Object
|
|
|
|
msgChannels []chan *Message
|
|
mux sync.Mutex
|
|
}
|
|
|
|
// Instance returns the underlying GstBus instance.
|
|
func (b *Bus) Instance() *C.GstBus { return C.toGstBus(b.unsafe()) }
|
|
|
|
func (b *Bus) deliverMessages() {
|
|
for {
|
|
msg := b.BlockPopMessage()
|
|
if msg == nil {
|
|
return
|
|
}
|
|
b.mux.Lock()
|
|
for _, ch := range b.msgChannels {
|
|
ch <- msg.Ref()
|
|
}
|
|
b.mux.Unlock()
|
|
msg.Unref()
|
|
}
|
|
}
|
|
|
|
// MessageChan returns a new channel to listen for messages asynchronously. Messages
|
|
// should be unreffed after each usage. Messages are delivered to channels in the
|
|
// order in which this function was called.
|
|
//
|
|
// While a message is being delivered to created channels, there is a lock on creating
|
|
// new ones.
|
|
func (b *Bus) MessageChan() chan *Message {
|
|
b.mux.Lock()
|
|
defer b.mux.Unlock()
|
|
ch := make(chan *Message)
|
|
b.msgChannels = append(b.msgChannels, ch)
|
|
if len(b.msgChannels) == 1 {
|
|
go b.deliverMessages()
|
|
}
|
|
return ch
|
|
}
|
|
|
|
// PopMessage attempts to pop a message from the bus. It returns nil if none are available.
|
|
// The message should be unreffed after usage.
|
|
func (b *Bus) PopMessage(timeout int) *Message {
|
|
if b.Instance() == nil {
|
|
return nil
|
|
}
|
|
dur := time.Duration(timeout) * time.Second
|
|
cTimeout := C.GstClockTime(dur.Nanoseconds())
|
|
msg := C.gst_bus_timed_pop(b.Instance(), cTimeout)
|
|
if msg == nil {
|
|
return nil
|
|
}
|
|
return wrapMessage(msg)
|
|
}
|
|
|
|
// BlockPopMessage blocks until a message is available on the bus and then returns it.
|
|
// This function can return nil if the bus is closed. The message should be unreffed
|
|
// after usage.
|
|
func (b *Bus) BlockPopMessage() *Message {
|
|
for {
|
|
if b.Instance() == nil {
|
|
return nil
|
|
}
|
|
msg := b.PopMessage(1)
|
|
if msg == nil {
|
|
continue
|
|
}
|
|
return msg
|
|
}
|
|
}
|