62 lines
1.4 KiB
Go
62 lines
1.4 KiB
Go
|
// 23 february 2015
|
||
|
|
||
|
package ui
|
||
|
|
||
|
// Go pointers cannot be passed to C safely.
|
||
|
// The Go runtime will move them around, tripping C up.
|
||
|
// However, C pointers don't have this problem.
|
||
|
// Solution: use the C pointer to get the Go object needed!
|
||
|
|
||
|
import (
|
||
|
"fmt"
|
||
|
"sync"
|
||
|
)
|
||
|
|
||
|
var (
|
||
|
ids map[uintptr]interface{}
|
||
|
idsLock sync.Mutex
|
||
|
)
|
||
|
|
||
|
func getGoObject(cptr uintptr) interface{} {
|
||
|
idsLock.Lock()
|
||
|
defer idsLock.Unlock()
|
||
|
|
||
|
if i, ok := ids[cptr]; ok {
|
||
|
return i
|
||
|
}
|
||
|
panic(fmt.Errorf("[BUG in package ui; report to andlabs immediately] C pointer %p not associated with Go object in package ui", cptr))
|
||
|
}
|
||
|
|
||
|
func associateGoObject(cptr uintptr, goobj interface{}) {
|
||
|
idsLock.Lock()
|
||
|
defer idsLock.Unlock()
|
||
|
|
||
|
if i, ok := ids[cptr]; ok {
|
||
|
panic(fmt.Errorf("[BUG in package ui; report to andlabs immediately] C pointer %p already associated with Go object of type %T but we want to associate it with Go object of type %T", cptr, i, goobj))
|
||
|
}
|
||
|
ids[cptr] = goobj
|
||
|
}
|
||
|
|
||
|
func disassociateGoObject(cptr uintptr) {
|
||
|
idsLock.Lock()
|
||
|
defer idsLock.Unlock()
|
||
|
|
||
|
if _, ok := ids[cptr]; !ok {
|
||
|
panic(fmt.Errorf("[BUG in package ui; report to andlabs immediately] C pointer %p not associated with any Go object but we want to disassociate it", cptr))
|
||
|
}
|
||
|
delete(ids, cptr)
|
||
|
}
|
||
|
|
||
|
func idsHandler(req *request) bool {
|
||
|
switch req.id {
|
||
|
case reqObjectDestroyed:
|
||
|
disassociateGoObject(req.ptr)
|
||
|
return true
|
||
|
}
|
||
|
return false
|
||
|
}
|
||
|
|
||
|
func init() {
|
||
|
interopHandlers = append(interopHandlers, idsHandler)
|
||
|
}
|