Skip to content

Commit 0076a3a

Browse files
committed
add tests for dynvm
Signed-off-by: Protryon <[email protected]>
1 parent 70b1002 commit 0076a3a

File tree

11 files changed

+124
-41
lines changed

11 files changed

+124
-41
lines changed

.github/workflows/test.yml

+6
Original file line numberDiff line numberDiff line change
@@ -113,6 +113,12 @@ jobs:
113113
arch: x86_64
114114
action: test
115115
flags: --config=gcc
116+
- name: 'DynVM on Linux/x86_64'
117+
engine: 'null'
118+
os: ubuntu-20.04
119+
arch: x86_64
120+
action: test
121+
flags: --config=gcc
116122
- name: 'NullVM on Linux/x86_64 with ASan'
117123
engine: 'null'
118124
os: ubuntu-20.04

BUILD

+1
Original file line numberDiff line numberDiff line change
@@ -132,6 +132,7 @@ cc_library(
132132
"src/dyn/dyn_vm_plugin.cc",
133133
],
134134
hdrs = [
135+
"include/proxy-wasm/dyn.h",
135136
"include/proxy-wasm/dyn_vm.h",
136137
"include/proxy-wasm/dyn_vm_plugin.h",
137138
"include/proxy-wasm/wasm_api_impl.h",

bazel/wasm.bzl

+39
Original file line numberDiff line numberDiff line change
@@ -13,6 +13,7 @@
1313
# limitations under the License.
1414

1515
load("@rules_rust//rust:defs.bzl", "rust_binary")
16+
load("@rules_rust//rust:defs.bzl", "rust_shared_library")
1617

1718
def _wasm_rust_transition_impl(settings, attr):
1819
return {
@@ -59,6 +60,17 @@ def _wasm_binary_impl(ctx):
5960

6061
return [DefaultInfo(files = depset([out]), runfiles = ctx.runfiles([out]))]
6162

63+
def _dyn_binary_impl(ctx):
64+
out = ctx.actions.declare_file(ctx.label.name)
65+
ctx.actions.run(
66+
executable = "cp",
67+
arguments = [ctx.files.binary[0].path, out.path],
68+
outputs = [out],
69+
inputs = ctx.files.binary,
70+
)
71+
72+
return [DefaultInfo(files = depset([out]), runfiles = ctx.runfiles([out]))]
73+
6274
def _wasm_attrs(transition):
6375
return {
6476
"binary": attr.label(mandatory = True, cfg = transition),
@@ -67,6 +79,11 @@ def _wasm_attrs(transition):
6779
"_whitelist_function_transition": attr.label(default = "@bazel_tools//tools/whitelists/function_transition_whitelist"),
6880
}
6981

82+
def _dyn_attrs():
83+
return {
84+
"binary": attr.label(mandatory = True),
85+
}
86+
7087
wasm_rust_binary_rule = rule(
7188
implementation = _wasm_binary_impl,
7289
attrs = _wasm_attrs(wasm_rust_transition),
@@ -77,6 +94,11 @@ wasi_rust_binary_rule = rule(
7794
attrs = _wasm_attrs(wasi_rust_transition),
7895
)
7996

97+
dyn_rust_binary_rule = rule(
98+
implementation = _dyn_binary_impl,
99+
attrs = _dyn_attrs(),
100+
)
101+
80102
def wasm_rust_binary(name, tags = [], wasi = False, signing_key = [], **kwargs):
81103
wasm_name = "_wasm_" + name.replace(".", "_")
82104
kwargs.setdefault("visibility", ["//visibility:public"])
@@ -100,3 +122,20 @@ def wasm_rust_binary(name, tags = [], wasi = False, signing_key = [], **kwargs):
100122
signing_key = signing_key,
101123
tags = tags + ["manual"],
102124
)
125+
126+
def dyn_rust_library(name, tags = [], **kwargs):
127+
dyn_name = "_dyn_" + name.replace(".", "_")
128+
kwargs.setdefault("visibility", ["//visibility:public"])
129+
130+
rust_shared_library(
131+
name = dyn_name,
132+
edition = "2018",
133+
tags = ["manual"],
134+
**kwargs
135+
)
136+
137+
dyn_rust_binary_rule(
138+
name = name,
139+
binary = ":" + dyn_name,
140+
tags = tags + ["manual"],
141+
)

include/proxy-wasm/dyn_vm.h

-3
Original file line numberDiff line numberDiff line change
@@ -24,9 +24,6 @@
2424

2525
namespace proxy_wasm {
2626

27-
class WasmVm;
28-
std::unique_ptr<WasmVm> createDynVm();
29-
3027
// The DynVm wraps a C++ Wasm plugin which has been compiled with the Wasm API
3128
// and dynamically linked into the proxy.
3229
struct DynVm : public WasmVm {

src/dyn/dyn.cc

+1
Original file line numberDiff line numberDiff line change
@@ -13,6 +13,7 @@
1313
// See the License for the specific language governing permissions and
1414
// limitations under the License.
1515

16+
#include "include/proxy-wasm/dyn.h"
1617
#include "include/proxy-wasm/dyn_vm.h"
1718

1819
namespace proxy_wasm {

src/dyn/dyn_vm.cc

+10-10
Original file line numberDiff line numberDiff line change
@@ -16,16 +16,15 @@
1616
#include "include/proxy-wasm/dyn_vm.h"
1717

1818
#include <cstring>
19-
19+
#include <dlfcn.h>
2020
#include <limits>
2121
#include <memory>
22+
#include <sys/mman.h>
23+
#include <sys/syscall.h>
24+
#include <unistd.h>
2225
#include <unordered_map>
2326
#include <utility>
2427
#include <vector>
25-
#include <sys/mman.h>
26-
#include <dlfcn.h>
27-
#include <unistd.h>
28-
#include <sys/syscall.h>
2928

3029
namespace proxy_wasm {
3130

@@ -42,7 +41,7 @@ std::unique_ptr<WasmVm> DynVm::clone() {
4241
}
4342

4443
// "Load" the plugin by obtaining a pointer to it from the factory.
45-
bool DynVm::load(std::string_view shared_lib, std::string_view /*precompiled*/,
44+
bool DynVm::load(std::string_view plugin_name, std::string_view /*precompiled*/,
4645
const std::unordered_map<uint32_t, std::string> & /*function_names*/) {
4746
plugin_ = std::make_unique<DynVmPlugin>();
4847
plugin_->source = std::make_shared<DynVmPluginSource>();
@@ -59,13 +58,14 @@ bool DynVm::load(std::string_view shared_lib, std::string_view /*precompiled*/,
5958

6059
size_t written = 0;
6160
ssize_t wrote;
62-
const char *data = shared_lib.data();
63-
while (written < shared_lib.size()) {
64-
wrote = write(plugin_->source->memfd, data + written, shared_lib.length() - written);
61+
const char *data = plugin_name.data();
62+
while (written < plugin_name.size()) {
63+
wrote = write(plugin_->source->memfd, data + written, plugin_name.length() - written);
6564
if (wrote < 0) {
6665
integration()->error("failed to write to memfd: " + std::string(strerror(errno)));
6766
return false;
68-
} else if (wrote == 0) {
67+
}
68+
if (wrote == 0) {
6969
integration()->error("failed to write to memfd, EOF on write");
7070
return false;
7171
}

src/dyn/dyn_vm_plugin.cc

+28-28
Original file line numberDiff line numberDiff line change
@@ -16,14 +16,14 @@
1616
#include "include/proxy-wasm/wasm_vm.h"
1717

1818
#include "include/proxy-wasm/dyn_vm_plugin.h"
19-
#include <iostream>
19+
#include <cstdarg>
2020
#include <dlfcn.h>
21-
#include <thread>
21+
#include <iostream>
2222
#include <sstream>
23-
#include <string>
2423
#include <stdexcept>
24+
#include <string>
25+
#include <thread>
2526
#include <unistd.h>
26-
#include <cstdarg>
2727

2828
namespace proxy_wasm {
2929

@@ -61,16 +61,16 @@ DynVmPluginSource::~DynVmPluginSource() {
6161
}
6262

6363
void DynVmPlugin::getFunction(std::string_view function_name, proxy_wasm::WasmCallVoid<0> *f) {
64-
if (source->dl_handle == NULL) {
64+
if (source->dl_handle == nullptr) {
6565
*f = nullptr;
6666
return;
6767
}
6868
void *target = dlsym(source->dl_handle, std::string(function_name).c_str());
69-
if (target == NULL) {
69+
if (target == nullptr) {
7070
*f = nullptr;
7171
return;
7272
}
73-
void (*target_func)() = reinterpret_cast<void (*)()>(target);
73+
auto target_func = reinterpret_cast<void (*)()>(target);
7474
*f = [this, target_func, function_name](proxy_wasm::ContextBase *context) {
7575
proxy_wasm::SaveRestoreContext saved_context(context);
7676
wasm_vm_->integration()->trace(call_format(function_name, 0));
@@ -79,16 +79,16 @@ void DynVmPlugin::getFunction(std::string_view function_name, proxy_wasm::WasmCa
7979
}
8080

8181
void DynVmPlugin::getFunction(std::string_view function_name, proxy_wasm::WasmCallVoid<1> *f) {
82-
if (source->dl_handle == NULL) {
82+
if (source->dl_handle == nullptr) {
8383
*f = nullptr;
8484
return;
8585
}
8686
void *target = dlsym(source->dl_handle, std::string(function_name).c_str());
87-
if (target == NULL) {
87+
if (target == nullptr) {
8888
*f = nullptr;
8989
return;
9090
}
91-
void (*target_func)(uint64_t) = reinterpret_cast<void (*)(uint64_t)>(target);
91+
auto target_func = reinterpret_cast<void (*)(uint64_t)>(target);
9292
*f = [this, target_func, function_name](proxy_wasm::ContextBase *context, proxy_wasm::Word w1) {
9393
proxy_wasm::SaveRestoreContext saved_context(context);
9494
wasm_vm_->integration()->trace(call_format(function_name, 1, w1));
@@ -97,16 +97,16 @@ void DynVmPlugin::getFunction(std::string_view function_name, proxy_wasm::WasmCa
9797
}
9898

9999
void DynVmPlugin::getFunction(std::string_view function_name, proxy_wasm::WasmCallVoid<2> *f) {
100-
if (source->dl_handle == NULL) {
100+
if (source->dl_handle == nullptr) {
101101
*f = nullptr;
102102
return;
103103
}
104104
void *target = dlsym(source->dl_handle, std::string(function_name).c_str());
105-
if (target == NULL) {
105+
if (target == nullptr) {
106106
*f = nullptr;
107107
return;
108108
}
109-
void (*target_func)(uint64_t, uint64_t) = reinterpret_cast<void (*)(uint64_t, uint64_t)>(target);
109+
auto target_func = reinterpret_cast<void (*)(uint64_t, uint64_t)>(target);
110110
*f = [this, target_func, function_name](proxy_wasm::ContextBase *context, proxy_wasm::Word w1,
111111
proxy_wasm::Word w2) {
112112
proxy_wasm::SaveRestoreContext saved_context(context);
@@ -116,16 +116,16 @@ void DynVmPlugin::getFunction(std::string_view function_name, proxy_wasm::WasmCa
116116
}
117117

118118
void DynVmPlugin::getFunction(std::string_view function_name, proxy_wasm::WasmCallVoid<3> *f) {
119-
if (source->dl_handle == NULL) {
119+
if (source->dl_handle == nullptr) {
120120
*f = nullptr;
121121
return;
122122
}
123123
void *target = dlsym(source->dl_handle, std::string(function_name).c_str());
124-
if (target == NULL) {
124+
if (target == nullptr) {
125125
*f = nullptr;
126126
return;
127127
}
128-
void (*target_func)(uint64_t, uint64_t, uint64_t) =
128+
auto target_func =
129129
reinterpret_cast<void (*)(uint64_t, uint64_t, uint64_t)>(target);
130130
*f = [this, target_func, function_name](proxy_wasm::ContextBase *context, proxy_wasm::Word w1,
131131
proxy_wasm::Word w2, proxy_wasm::Word w3) {
@@ -136,16 +136,16 @@ void DynVmPlugin::getFunction(std::string_view function_name, proxy_wasm::WasmCa
136136
}
137137

138138
void DynVmPlugin::getFunction(std::string_view function_name, proxy_wasm::WasmCallVoid<5> *f) {
139-
if (source->dl_handle == NULL) {
139+
if (source->dl_handle == nullptr) {
140140
*f = nullptr;
141141
return;
142142
}
143143
void *target = dlsym(source->dl_handle, std::string(function_name).c_str());
144-
if (target == NULL) {
144+
if (target == nullptr) {
145145
*f = nullptr;
146146
return;
147147
}
148-
void (*target_func)(uint64_t, uint64_t, uint64_t, uint64_t, uint64_t) =
148+
auto target_func =
149149
reinterpret_cast<void (*)(uint64_t, uint64_t, uint64_t, uint64_t, uint64_t)>(target);
150150
*f = [this, target_func, function_name](proxy_wasm::ContextBase *context, proxy_wasm::Word w1,
151151
proxy_wasm::Word w2, proxy_wasm::Word w3,
@@ -157,16 +157,16 @@ void DynVmPlugin::getFunction(std::string_view function_name, proxy_wasm::WasmCa
157157
}
158158

159159
void DynVmPlugin::getFunction(std::string_view function_name, proxy_wasm::WasmCallWord<1> *f) {
160-
if (source->dl_handle == NULL || function_name == "malloc") {
160+
if (source->dl_handle == nullptr || function_name == "malloc") {
161161
*f = nullptr;
162162
return;
163163
}
164164
void *target = dlsym(source->dl_handle, std::string(function_name).c_str());
165-
if (target == NULL) {
165+
if (target == nullptr) {
166166
*f = nullptr;
167167
return;
168168
}
169-
uint64_t (*target_func)(uint64_t) = reinterpret_cast<uint64_t (*)(uint64_t)>(target);
169+
auto target_func = reinterpret_cast<uint64_t (*)(uint64_t)>(target);
170170
*f = [this, target_func, function_name](proxy_wasm::ContextBase *context, proxy_wasm::Word w1) {
171171
proxy_wasm::SaveRestoreContext saved_context(context);
172172
wasm_vm_->integration()->trace(call_format(function_name, 1, w1));
@@ -175,16 +175,16 @@ void DynVmPlugin::getFunction(std::string_view function_name, proxy_wasm::WasmCa
175175
}
176176

177177
void DynVmPlugin::getFunction(std::string_view function_name, proxy_wasm::WasmCallWord<2> *f) {
178-
if (source->dl_handle == NULL) {
178+
if (source->dl_handle == nullptr) {
179179
*f = nullptr;
180180
return;
181181
}
182182
void *target = dlsym(source->dl_handle, std::string(function_name).c_str());
183-
if (target == NULL) {
183+
if (target == nullptr) {
184184
*f = nullptr;
185185
return;
186186
}
187-
uint64_t (*target_func)(uint64_t, uint64_t) =
187+
auto target_func =
188188
reinterpret_cast<uint64_t (*)(uint64_t, uint64_t)>(target);
189189
*f = [this, target_func, function_name](proxy_wasm::ContextBase *context, proxy_wasm::Word w1,
190190
proxy_wasm::Word w2) {
@@ -195,16 +195,16 @@ void DynVmPlugin::getFunction(std::string_view function_name, proxy_wasm::WasmCa
195195
}
196196

197197
void DynVmPlugin::getFunction(std::string_view function_name, proxy_wasm::WasmCallWord<3> *f) {
198-
if (source->dl_handle == NULL) {
198+
if (source->dl_handle == nullptr) {
199199
*f = nullptr;
200200
return;
201201
}
202202
void *target = dlsym(source->dl_handle, std::string(function_name).c_str());
203-
if (target == NULL) {
203+
if (target == nullptr) {
204204
*f = nullptr;
205205
return;
206206
}
207-
uint64_t (*target_func)(uint64_t, uint64_t, uint64_t) =
207+
auto target_func =
208208
reinterpret_cast<uint64_t (*)(uint64_t, uint64_t, uint64_t)>(target);
209209
*f = [this, target_func, function_name](proxy_wasm::ContextBase *context, proxy_wasm::Word w1,
210210
proxy_wasm::Word w2, proxy_wasm::Word w3) {

test/BUILD

+15
Original file line numberDiff line numberDiff line change
@@ -180,6 +180,21 @@ cc_test(
180180
],
181181
)
182182

183+
cc_test(
184+
name = "dyn_vm_test",
185+
srcs = ["dyn_vm_test.cc"],
186+
data = [
187+
"//test/test_data:abi_export.so",
188+
],
189+
linkstatic = 1,
190+
deps = [
191+
":utility_lib",
192+
"//:lib",
193+
"@com_google_googletest//:gtest",
194+
"@com_google_googletest//:gtest_main",
195+
],
196+
)
197+
183198
cc_test(
184199
name = "wasm_vm_test",
185200
timeout = "long",

test/test_data/BUILD

+6
Original file line numberDiff line numberDiff line change
@@ -13,6 +13,7 @@
1313
# limitations under the License.
1414

1515
load("@proxy_wasm_cpp_host//bazel:wasm.bzl", "wasm_rust_binary")
16+
load("@proxy_wasm_cpp_host//bazel:wasm.bzl", "dyn_rust_library")
1617
load("@proxy_wasm_cpp_sdk//bazel:defs.bzl", "proxy_wasm_cc_binary")
1718

1819
licenses(["notice"]) # Apache 2
@@ -24,6 +25,11 @@ wasm_rust_binary(
2425
srcs = ["abi_export.rs"],
2526
)
2627

28+
dyn_rust_library(
29+
name = "abi_export.so",
30+
srcs = ["abi_export.rs"],
31+
)
32+
2733
wasm_rust_binary(
2834
name = "abi_export.signed.with.key1.wasm",
2935
srcs = ["abi_export.rs"],

test/utility.cc

+11
Original file line numberDiff line numberDiff line change
@@ -41,6 +41,17 @@ std::vector<std::string> getWasmEngines() {
4141
return engines;
4242
}
4343

44+
std::vector<std::string> getDynEngines() {
45+
std::vector<std::string> engines = {
46+
#if defined(PROXY_WASM_HOST_ENGINE_DYN)
47+
"dyn",
48+
#endif
49+
""
50+
};
51+
engines.pop_back();
52+
return engines;
53+
}
54+
4455
std::string readTestWasmFile(const std::string &filename) {
4556
auto path = "test/test_data/" + filename;
4657
std::ifstream file(path, std::ios::binary);

0 commit comments

Comments
 (0)