threads.cc 3.37 KB
Newer Older
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21
// Copyright 2019 the V8 project authors. All rights reserved.
// Use of this source code is governed by a BSD-style license that can be
// found in the LICENSE file.

#include "test/wasm-api-tests/wasm-api-test.h"

#include <mutex>
#include <thread>

namespace v8 {
namespace internal {
namespace wasm {

using ::wasm::Shared;

namespace {

const int kNumThreads = 10;
const int kIterationsPerThread = 3;
int g_traces;

22
own<Trap> Callback(void* env, const Val args[], Val results[]) {
23 24 25 26 27 28
  std::lock_guard<std::mutex> lock(*reinterpret_cast<std::mutex*>(env));
  g_traces += args[0].i32();
  return nullptr;
}

void Main(Engine* engine, Shared<Module>* shared, std::mutex* mutex, int id) {
29 30
  own<Store> store = Store::make(engine);
  own<Module> module = Module::obtain(store.get(), shared);
31 32 33 34 35
  EXPECT_NE(nullptr, module.get());
  for (int i = 0; i < kIterationsPerThread; i++) {
    std::this_thread::sleep_for(std::chrono::microseconds(100));

    // Create imports.
36 37 38 39 40
    own<FuncType> func_type =
        FuncType::make(ownvec<ValType>::make(ValType::make(::wasm::I32)),
                       ownvec<ValType>::make());
    own<Func> func = Func::make(store.get(), func_type.get(), Callback, mutex);
    own<::wasm::GlobalType> global_type =
41
        ::wasm::GlobalType::make(ValType::make(::wasm::I32), ::wasm::CONST);
42
    own<Global> global =
43 44 45 46 47 48 49
        Global::make(store.get(), global_type.get(), Val::i32(id));

    // Instantiate and run.
    // With the current implementation of the WasmModuleBuilder, global
    // imports always come before function imports, regardless of the
    // order of builder()->Add*Import() calls below.
    Extern* imports[] = {global.get(), func.get()};
50 51
    own<Instance> instance = Instance::make(store.get(), module.get(), imports);
    ownvec<Extern> exports = instance->exports();
52 53 54 55 56 57 58 59 60 61 62 63
    Func* run_func = exports[0]->func();
    run_func->call();
  }
}

}  // namespace

TEST_F(WasmCapiTest, Threads) {
  // Create module.
  ValueType i32_type[] = {kWasmI32};
  FunctionSig param_i32(0, 1, i32_type);
  uint32_t callback_index =
64
      builder()->AddImport(base::CStrVector("callback"), &param_i32);
65
  uint32_t global_index =
66
      builder()->AddGlobalImport(base::CStrVector("id"), kWasmI32, false);
67 68

  byte code[] = {
69
      WASM_CALL_FUNCTION(callback_index, WASM_GLOBAL_GET(global_index))};
70
  FunctionSig empty_sig(0, 0, nullptr);
71
  AddExportedFunction(base::CStrVector("run"), code, sizeof(code), &empty_sig);
72
  Compile();
73
  own<Shared<Module>> shared = module()->share();
74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92

  // Spawn threads.
  g_traces = 0;
  std::mutex mutex;
  std::thread threads[kNumThreads];
  for (int i = 0; i < kNumThreads; i++) {
    threads[i] = std::thread(Main, engine(), shared.get(), &mutex, i);
  }
  for (int i = 0; i < kNumThreads; i++) {
    threads[i].join();
  }
  // Each thread in each iteration adds its ID to {traces}, so in the end
  // we expect kIterationsPerThread * sum([0, ..., kNumThreads-1]).
  // Per Gauss:
  const int kExpected =
      kIterationsPerThread * (kNumThreads - 1) * kNumThreads / 2;
  EXPECT_EQ(kExpected, g_traces);
}

93 94
TEST_F(WasmCapiTest, MultiStoresOneThread) {
  // These Stores intentionally have overlapping, but non-nested lifetimes.
95 96 97
  own<Store> store1 = Store::make(engine());
  own<Store> store2 = Store::make(engine());
  own<Store> store3 = Store::make(engine());
98 99 100 101 102
  store1.reset();
  store2.reset();
  store3.reset();
}

103 104 105
}  // namespace wasm
}  // namespace internal
}  // namespace v8