Skip to content

Commit fd31630

Browse files
authored
Merge pull request #128 from Code-Hex/fix/124-2
fixed issue #124 again
2 parents 6c8b0b0 + a8cf290 commit fd31630

6 files changed

+100
-24
lines changed

Makefile

+8
Original file line numberDiff line numberDiff line change
@@ -12,6 +12,14 @@ fmt:
1212
test:
1313
go test -exec "go run $(PWD)/cmd/codesign" ./... -timeout 60s -v
1414

15+
.PHONY: test/run
16+
test/run:
17+
go test -exec "go run $(PWD)/cmd/codesign" ./... -timeout 5m -v -run $(TARGET)
18+
19+
.PHONY: test/run/124
20+
test/run/124:
21+
TEST_ISSUE_124=1 $(MAKE) test/run TARGET=TestRunIssue124
22+
1523
.PHONY: download_kernel
1624
download_kernel:
1725
curl --output-dir testdata -LO $(KERNEL_DOWNLOAD_URL)

virtualization.go

+12-22
Original file line numberDiff line numberDiff line change
@@ -75,7 +75,7 @@ type VirtualMachine struct {
7575

7676
*pointer
7777
dispatchQueue unsafe.Pointer
78-
stateHandle cgo.Handle
78+
stateHandle *machineState
7979

8080
finalizeOnce sync.Once
8181
}
@@ -103,18 +103,19 @@ func NewVirtualMachine(config *VirtualMachineConfiguration) (*VirtualMachine, er
103103
cs := (*char)(objc.GetUUID())
104104
dispatchQueue := C.makeDispatchQueue(cs.CString())
105105

106-
stateHandle := cgo.NewHandle(&machineState{
106+
stateHandle := &machineState{
107107
state: VirtualMachineState(0),
108108
stateNotify: infinity.NewChannel[VirtualMachineState](),
109-
})
109+
}
110110

111+
stateHandlePtr := cgo.NewHandle(stateHandle)
111112
v := &VirtualMachine{
112113
id: cs.String(),
113114
pointer: objc.NewPointer(
114115
C.newVZVirtualMachineWithDispatchQueue(
115116
objc.Ptr(config),
116117
dispatchQueue,
117-
unsafe.Pointer(&stateHandle),
118+
unsafe.Pointer(&stateHandlePtr),
118119
),
119120
),
120121
dispatchQueue: dispatchQueue,
@@ -123,6 +124,7 @@ func NewVirtualMachine(config *VirtualMachineConfiguration) (*VirtualMachine, er
123124

124125
objc.SetFinalizer(v, func(self *VirtualMachine) {
125126
self.finalize()
127+
stateHandlePtr.Delete()
126128
})
127129
return v, nil
128130
}
@@ -165,30 +167,18 @@ func changeStateOnObserver(newStateRaw C.int, cgoHandlerPtr unsafe.Pointer) {
165167
v.mu.Unlock()
166168
}
167169

168-
//export deleteStateHandler
169-
func deleteStateHandler(cgoHandlerPtr unsafe.Pointer) {
170-
stateHandler := *(*cgo.Handle)(cgoHandlerPtr)
171-
stateHandler.Delete()
172-
}
173-
174170
// State represents execution state of the virtual machine.
175171
func (v *VirtualMachine) State() VirtualMachineState {
176-
// I expected it will not cause panic.
177-
// if caused panic, that's unexpected behavior.
178-
val, _ := v.stateHandle.Value().(*machineState)
179-
val.mu.RLock()
180-
defer val.mu.RUnlock()
181-
return val.state
172+
v.stateHandle.mu.RLock()
173+
defer v.stateHandle.mu.RUnlock()
174+
return v.stateHandle.state
182175
}
183176

184177
// StateChangedNotify gets notification is changed execution state of the virtual machine.
185178
func (v *VirtualMachine) StateChangedNotify() <-chan VirtualMachineState {
186-
// I expected it will not cause panic.
187-
// if caused panic, that's unexpected behavior.
188-
val, _ := v.stateHandle.Value().(*machineState)
189-
val.mu.RLock()
190-
defer val.mu.RUnlock()
191-
return val.stateNotify.Out()
179+
v.stateHandle.mu.RLock()
180+
defer v.stateHandle.mu.RUnlock()
181+
return v.stateHandle.stateNotify.Out()
192182
}
193183

194184
// CanStart returns true if the machine is in a state that can be started.

virtualization_11.h

-1
Original file line numberDiff line numberDiff line change
@@ -13,7 +13,6 @@
1313
void connectionHandler(void *connection, void *err, void *cgoHandlerPtr);
1414
void changeStateOnObserver(int state, void *cgoHandler);
1515
bool shouldAcceptNewConnectionHandler(void *cgoHandler, void *connection, void *socketDevice);
16-
void deleteStateHandler(void *cgoHandlerPtr);
1716

1817
@interface Observer : NSObject
1918
- (void)observeValueForKeyPath:(NSString *)keyPath ofObject:(id)object change:(NSDictionary *)change context:(void *)context;

virtualization_11.m

-1
Original file line numberDiff line numberDiff line change
@@ -39,7 +39,6 @@ - (void)dealloc
3939
[self removeObserver:_observer forKeyPath:@"state"];
4040
[_observer release];
4141
[super dealloc];
42-
deleteStateHandler(_stateHandler);
4342
}
4443
@end
4544

virtualization_test.go

+69
Original file line numberDiff line numberDiff line change
@@ -4,6 +4,7 @@ import (
44
"errors"
55
"fmt"
66
"os"
7+
"runtime"
78
"syscall"
89
"testing"
910
"time"
@@ -342,3 +343,71 @@ func TestVirtualMachineStateString(t *testing.T) {
342343
}
343344
}
344345
}
346+
347+
func TestRunIssue124(t *testing.T) {
348+
if os.Getenv("TEST_ISSUE_124") != "1" {
349+
t.Skip()
350+
}
351+
container := newVirtualizationMachine(t,
352+
func(vmc *vz.VirtualMachineConfiguration) error {
353+
return setupConsoleConfig(vmc)
354+
},
355+
)
356+
defer container.Close()
357+
358+
sshSession := container.NewSession(t)
359+
defer sshSession.Close()
360+
361+
vm := container.VirtualMachine
362+
363+
if got := vm.State(); vz.VirtualMachineStateRunning != got {
364+
t.Fatalf("want state %v but got %v", vz.VirtualMachineStateRunning, got)
365+
}
366+
if got := vm.CanPause(); !got {
367+
t.Fatal("want CanPause is true")
368+
}
369+
if err := vm.Pause(); err != nil {
370+
t.Fatal(err)
371+
}
372+
373+
timeout := 5 * time.Second
374+
waitState(t, timeout, vm, vz.VirtualMachineStatePausing)
375+
waitState(t, timeout, vm, vz.VirtualMachineStatePaused)
376+
377+
if got := vm.State(); vz.VirtualMachineStatePaused != got {
378+
t.Fatalf("want state %v but got %v", vz.VirtualMachineStatePaused, got)
379+
}
380+
if got := vm.CanResume(); !got {
381+
t.Fatal("want CanPause is true")
382+
}
383+
if err := vm.Resume(); err != nil {
384+
t.Fatal(err)
385+
}
386+
387+
waitState(t, timeout, vm, vz.VirtualMachineStateResuming)
388+
waitState(t, timeout, vm, vz.VirtualMachineStateRunning)
389+
390+
if got := vm.CanRequestStop(); !got {
391+
t.Fatal("want CanRequestStop is true")
392+
}
393+
394+
ch := make(chan bool)
395+
vm.SetMachineStateFinalizer(func() {
396+
ch <- true
397+
})
398+
399+
runtime.GC()
400+
select {
401+
case <-ch:
402+
t.Errorf("expected finalizer do not run")
403+
case <-time.After(4 * time.Minute):
404+
}
405+
406+
runtime.GC()
407+
sshSession.Run("poweroff")
408+
waitState(t, timeout, vm, vz.VirtualMachineStateStopped)
409+
410+
if got := vm.State(); vz.VirtualMachineStateStopped != got {
411+
t.Fatalf("want state %v but got %v", vz.VirtualMachineStateStopped, got)
412+
}
413+
}

vz_export_test.go

+11
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,11 @@
1+
package vz
2+
3+
import (
4+
"runtime"
5+
)
6+
7+
func (v *VirtualMachine) SetMachineStateFinalizer(f func()) {
8+
runtime.SetFinalizer(v.stateHandle, func(self *machineState) {
9+
f()
10+
})
11+
}

0 commit comments

Comments
 (0)