package repeater import ( "errors" "fmt" "log" "github.com/pd0mz/go-dmr/bptc" "github.com/pd0mz/go-dmr/dmr" "github.com/pd0mz/go-dmr/ipsc" ) var voiceFrameMap = map[uint16]uint8{ ipsc.VoiceDataA: 0, ipsc.VoiceDataB: 1, ipsc.VoiceDataC: 2, ipsc.VoiceDataD: 3, ipsc.VoiceDataE: 4, ipsc.VoiceDataF: 5, } func (r *Repeater) HandleVoiceData(p *ipsc.Packet) error { r.DataCallEnd(p) var ( err error payload = make([]byte, 12) ) if err = bptc.Process(dmr.ExtractInfoBits(p.PayloadBits), payload); err != nil { return err } if r.Slot[p.Timeslot].State != VoiceCallRunning { r.VoiceCallStart(p) } return r.HandleVoiceFrame(p) } func (r *Repeater) HandleVoiceFrame(p *ipsc.Packet) error { // This may contain a sync frame sync := dmr.ExtractSyncBits(p.PayloadBits) patt := dmr.SyncPattern(sync) if patt != dmr.SyncPatternUnknown && r.Slot[p.Timeslot].Voice.Frame != 0 { fmt.Printf("sync pattern %s\n", dmr.SyncPatternName[patt]) r.Slot[p.Timeslot].Voice.Frame = 0 return nil } else { // This may be a duplicate frame var ( oldFrame = r.Slot[p.Timeslot].Voice.Frame newFrame = voiceFrameMap[p.SlotType] ) switch { case oldFrame > 5: // Ignore, wait for next sync frame return nil case newFrame == oldFrame: return errors.New("dmr/voice: ignored, duplicate frame") case newFrame > oldFrame: if newFrame-oldFrame > 1 { log.Printf("dmr/voice: framedrop, went from %c -> %c", 'A'+oldFrame, 'A'+newFrame) } case newFrame < oldFrame: if newFrame > 0 { log.Printf("dmr/voice: framedrop, went from %c -> %c", 'A'+oldFrame, 'A'+newFrame) } } r.Slot[p.Timeslot].Voice.Frame++ } // If it's not a sync frame, then it should have an EMB inside the sync field. var ( emb *dmr.EMB err error ) if emb, err = dmr.ParseEMB(dmr.ExtractEMBBitsFromSyncBits(sync)); err != nil { fmt.Println("unknown sync pattern, no EMB") return err } fmt.Printf("EMB LCSS %d\n", emb.LCSS) // TODO(maze): implement VBPTC matrix handling switch emb.LCSS { } return nil }