1
2
3
4
5 package main
6
7 import (
8 "fmt"
9 "log"
10 "path/filepath"
11 "plugin"
12 "strings"
13
14 "testplugin/common"
15 )
16
17 func init() {
18 common.X *= 5
19 }
20
21
22
23
24 func testUnnamed() {
25 p, err := plugin.Open("unnamed1.so")
26 if err != nil {
27 log.Fatalf(`plugin.Open("unnamed1.so"): %v`, err)
28 }
29 fn, err := p.Lookup("FuncInt")
30 if err != nil {
31 log.Fatalf(`unnamed1.so: Lookup("FuncInt") failed: %v`, err)
32 }
33 if got, want := fn.(func() int)(), 1; got != want {
34 log.Fatalf("unnamed1.so: FuncInt()=%d, want %d", got, want)
35 }
36
37 p, err = plugin.Open("unnamed2.so")
38 if err != nil {
39 log.Fatalf(`plugin.Open("unnamed2.so"): %v`, err)
40 }
41 fn, err = p.Lookup("FuncInt")
42 if err != nil {
43 log.Fatalf(`unnamed2.so: Lookup("FuncInt") failed: %v`, err)
44 }
45 if got, want := fn.(func() int)(), 2; got != want {
46 log.Fatalf("unnamed2.so: FuncInt()=%d, want %d", got, want)
47 }
48 }
49
50 func main() {
51 if got, want := common.X, 3*5; got != want {
52 log.Fatalf("before plugin load common.X=%d, want %d", got, want)
53 }
54
55 p, err := plugin.Open("plugin1.so")
56 if err != nil {
57 log.Fatalf("plugin.Open failed: %v", err)
58 }
59
60 const wantX = 3 * 5 * 7
61 if got := common.X; got != wantX {
62 log.Fatalf("after plugin load common.X=%d, want %d", got, wantX)
63 }
64
65 seven, err := p.Lookup("Seven")
66 if err != nil {
67 log.Fatalf(`Lookup("Seven") failed: %v`, err)
68 }
69 if got, want := *seven.(*int), 7; got != want {
70 log.Fatalf("plugin1.Seven=%d, want %d", got, want)
71 }
72
73 readFunc, err := p.Lookup("ReadCommonX")
74 if err != nil {
75 log.Fatalf(`plugin1.Lookup("ReadCommonX") failed: %v`, err)
76 }
77 if got := readFunc.(func() int)(); got != wantX {
78 log.Fatalf("plugin1.ReadCommonX()=%d, want %d", got, wantX)
79 }
80
81
82
83
84
85 subpPath, err := filepath.Abs("sub/plugin1.so")
86 if err != nil {
87 log.Fatalf("filepath.Abs(%q) failed: %v", subpPath, err)
88 }
89 subp, err := plugin.Open(subpPath)
90 if err != nil {
91 log.Fatalf("plugin.Open(%q) failed: %v", subpPath, err)
92 }
93
94 funcVar, err := subp.Lookup("FuncVar")
95 if err != nil {
96 log.Fatalf(`sub/plugin1.Lookup("FuncVar") failed: %v`, err)
97 }
98 called := false
99 *funcVar.(*func()) = func() {
100 called = true
101 }
102
103 readFunc, err = subp.Lookup("ReadCommonX")
104 if err != nil {
105 log.Fatalf(`sub/plugin1.Lookup("ReadCommonX") failed: %v`, err)
106 }
107 if got := readFunc.(func() int)(); got != wantX {
108 log.Fatalf("sub/plugin1.ReadCommonX()=%d, want %d", got, wantX)
109 }
110 if !called {
111 log.Fatal("calling ReadCommonX did not call FuncVar")
112 }
113
114 subf, err := subp.Lookup("F")
115 if err != nil {
116 log.Fatalf(`sub/plugin1.Lookup("F") failed: %v`, err)
117 }
118 if gotf := subf.(func() int)(); gotf != 17 {
119 log.Fatalf(`sub/plugin1.F()=%d, want 17`, gotf)
120 }
121 f, err := p.Lookup("F")
122 if err != nil {
123 log.Fatalf(`plugin1.Lookup("F") failed: %v`, err)
124 }
125 if gotf := f.(func() int)(); gotf != 3 {
126 log.Fatalf(`plugin1.F()=%d, want 17`, gotf)
127 }
128
129 p2, err := plugin.Open("plugin2.so")
130 if err != nil {
131 log.Fatalf("plugin.Open failed: %v", err)
132 }
133
134
135 if got, want := common.X, 2; got != want {
136 log.Fatalf("after loading plugin2, common.X=%d, want %d", got, want)
137 }
138
139 _, err = plugin.Open("plugin2-dup.so")
140 if err == nil {
141 log.Fatal(`plugin.Open("plugin2-dup.so"): duplicate open should have failed`)
142 }
143 if s := err.Error(); !strings.Contains(s, "already loaded") {
144 log.Fatal(`plugin.Open("plugin2.so"): error does not mention "already loaded"`)
145 }
146
147 _, err = plugin.Open("plugin-mismatch.so")
148 if err == nil {
149 log.Fatal(`plugin.Open("plugin-mismatch.so"): should have failed`)
150 }
151 if s := err.Error(); !strings.Contains(s, "different version") {
152 log.Fatalf(`plugin.Open("plugin-mismatch.so"): error does not mention "different version": %v`, s)
153 }
154
155 _, err = plugin.Open("plugin2-dup.so")
156 if err == nil {
157 log.Fatal(`plugin.Open("plugin2-dup.so"): duplicate open after bad plugin should have failed`)
158 }
159 _, err = plugin.Open("plugin2.so")
160 if err != nil {
161 log.Fatalf(`plugin.Open("plugin2.so"): second open with same name failed: %v`, err)
162 }
163
164
165
166
167
168 UnexportedNameReuse, _ := p.Lookup("UnexportedNameReuse")
169 UnexportedNameReuse.(func())()
170 UnexportedNameReuse, _ = p2.Lookup("UnexportedNameReuse")
171 UnexportedNameReuse.(func())()
172
173 testUnnamed()
174
175 fmt.Println("PASS")
176 }
177
View as plain text