Rename the refnum field, Num, to something much less likely to clash with an interface method set. Change-Id: If334966b2430f38118baded44461bd39298bafb0 Reviewed-on: https://go-review.googlesource.com/20983 Reviewed-by: Hyang-Ah Hana Kim <hyangah@gmail.com>
127 lines
2.8 KiB
Go
127 lines
2.8 KiB
Go
// Copyright 2014 The Go Authors. All rights reserved.
|
|
// Use of this source code is governed by a BSD-style
|
|
// license that can be found in the LICENSE file.
|
|
|
|
package seq
|
|
|
|
//#cgo LDFLAGS: -llog
|
|
//#include <android/log.h>
|
|
//#include <string.h>
|
|
//import "C"
|
|
|
|
import (
|
|
"fmt"
|
|
"runtime"
|
|
"sync"
|
|
)
|
|
|
|
type countedObj struct {
|
|
obj interface{}
|
|
cnt int32
|
|
}
|
|
|
|
// also known to bind/java/Seq.java and bind/objc/seq_darwin.m
|
|
const NullRefNum = 41
|
|
|
|
// refs stores Go objects that have been passed to another language.
|
|
var refs struct {
|
|
sync.Mutex
|
|
next int32 // next reference number to use for Go object, always negative
|
|
refs map[interface{}]int32
|
|
objs map[int32]countedObj
|
|
}
|
|
|
|
func init() {
|
|
refs.Lock()
|
|
refs.next = -24 // Go objects get negative reference numbers. Arbitrary starting point.
|
|
refs.refs = make(map[interface{}]int32)
|
|
refs.objs = make(map[int32]countedObj)
|
|
refs.Unlock()
|
|
}
|
|
|
|
// A Ref represents a Java or Go object passed across the language
|
|
// boundary.
|
|
type Ref struct {
|
|
Bind_Num int32
|
|
}
|
|
|
|
type proxy interface {
|
|
// Use a strange name and hope that user code does not implement it
|
|
Bind_proxy_refnum__() int32
|
|
}
|
|
|
|
// ToRefNum increments the reference count for an object and
|
|
// returns its refnum.
|
|
func ToRefNum(obj interface{}) int32 {
|
|
// We don't track foreign objects, so if obj is a proxy
|
|
// return its refnum.
|
|
if r, ok := obj.(proxy); ok {
|
|
refnum := r.Bind_proxy_refnum__()
|
|
if refnum <= 0 {
|
|
panic(fmt.Errorf("seq: proxy contained invalid Go refnum: %d", refnum))
|
|
}
|
|
return refnum
|
|
}
|
|
refs.Lock()
|
|
num := refs.refs[obj]
|
|
if num != 0 {
|
|
s := refs.objs[num]
|
|
refs.objs[num] = countedObj{s.obj, s.cnt + 1}
|
|
} else {
|
|
num = refs.next
|
|
refs.next--
|
|
if refs.next > 0 {
|
|
panic("refs.next underflow")
|
|
}
|
|
refs.refs[obj] = num
|
|
refs.objs[num] = countedObj{obj, 1}
|
|
}
|
|
refs.Unlock()
|
|
|
|
return int32(num)
|
|
}
|
|
|
|
// FromRefNum returns the Ref for a refnum. If the refnum specifies a
|
|
// foreign object, a finalizer is set to track its lifetime.
|
|
func FromRefNum(num int32) *Ref {
|
|
if num == NullRefNum {
|
|
return nil
|
|
}
|
|
ref := &Ref{num}
|
|
if ref.Bind_Num > 0 {
|
|
// This is a foreign object reference.
|
|
// Track its lifetime with a finalizer.
|
|
runtime.SetFinalizer(ref, FinalizeRef)
|
|
}
|
|
|
|
return ref
|
|
}
|
|
|
|
// Get returns the underlying object.
|
|
func (r *Ref) Get() interface{} {
|
|
refs.Lock()
|
|
o, ok := refs.objs[r.Bind_Num]
|
|
refs.Unlock()
|
|
if !ok {
|
|
panic(fmt.Sprintf("unknown ref %d", r.Bind_Num))
|
|
}
|
|
return o.obj
|
|
}
|
|
|
|
// Delete decrements the reference count and removes the pinned object
|
|
// from the object map when the reference count becomes zero.
|
|
func Delete(num int32) {
|
|
refs.Lock()
|
|
defer refs.Unlock()
|
|
o, ok := refs.objs[num]
|
|
if !ok {
|
|
panic(fmt.Sprintf("seq.Delete unknown refnum: %d", num))
|
|
}
|
|
if o.cnt <= 1 {
|
|
delete(refs.objs, num)
|
|
delete(refs.refs, o.obj)
|
|
} else {
|
|
refs.objs[num] = countedObj{o.obj, o.cnt - 1}
|
|
}
|
|
}
|