1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
base / native_library_fuchsia.cc [blame]
// Copyright 2018 The Chromium Authors
// Use of this source code is governed by a BSD-style license that can be
// found in the LICENSE file.
#include "base/native_library.h"
#include <fcntl.h>
#include <fuchsia/io/cpp/fidl.h>
#include <lib/fdio/directory.h>
#include <lib/fdio/io.h>
#include <lib/zx/vmo.h>
#include <stdio.h>
#include <zircon/dlfcn.h>
#include <zircon/status.h>
#include <zircon/syscalls.h>
#include <string_view>
#include "base/base_paths.h"
#include "base/files/file.h"
#include "base/files/file_path.h"
#include "base/fuchsia/fuchsia_logging.h"
#include "base/notreached.h"
#include "base/path_service.h"
#include "base/posix/safe_strerror.h"
#include "base/strings/strcat.h"
#include "base/strings/stringprintf.h"
#include "base/strings/utf_string_conversions.h"
#include "base/threading/thread_restrictions.h"
#include "base_paths.h"
namespace base {
std::string NativeLibraryLoadError::ToString() const {
return message;
}
NativeLibrary LoadNativeLibraryWithOptions(const FilePath& library_path,
const NativeLibraryOptions& options,
NativeLibraryLoadError* error) {
FilePath computed_path;
FilePath library_root_path =
base::PathService::CheckedGet(DIR_ASSETS).Append("lib");
if (library_path.IsAbsolute()) {
// See more info in fxbug.dev/105910.
if (!library_root_path.IsParent(library_path)) {
auto error_message =
base::StringPrintf("Absolute library paths must begin with %s",
library_root_path.value().c_str());
DLOG(ERROR) << error_message;
if (error) {
error->message = std::move(error_message);
}
return nullptr;
}
computed_path = library_path;
} else {
computed_path = library_root_path.Append(library_path);
}
// Use fdio_open3_fd (a Fuchsia-specific API) here so we can pass the
// appropriate FS rights flags to request executability.
// TODO(crbug.com/40655456): Teach base::File about FLAG_WIN_EXECUTE on
// Fuchsia, and then use it here instead of using fdio_open3_fd() directly.
base::ScopedFD fd;
zx_status_t status =
fdio_open3_fd(computed_path.value().c_str(),
static_cast<uint64_t>(fuchsia::io::PERM_READABLE |
fuchsia::io::PERM_EXECUTABLE),
base::ScopedFD::Receiver(fd).get());
if (status != ZX_OK) {
if (error) {
error->message =
base::StringPrintf("fdio_open_fd: %s", zx_status_get_string(status));
}
return nullptr;
}
zx::vmo vmo;
status = fdio_get_vmo_exec(fd.get(), vmo.reset_and_get_address());
if (status != ZX_OK) {
if (error) {
error->message = base::StringPrintf("fdio_get_vmo_exec: %s",
zx_status_get_string(status));
}
return nullptr;
}
NativeLibrary result = dlopen_vmo(vmo.get(), RTLD_LAZY | RTLD_LOCAL);
return result;
}
void UnloadNativeLibrary(NativeLibrary library) {
// dlclose() is a no-op on Fuchsia, so do nothing here.
}
void* GetFunctionPointerFromNativeLibrary(NativeLibrary library,
const char* name) {
return dlsym(library, name);
}
std::string GetNativeLibraryName(std::string_view name) {
return StrCat({"lib", name, ".so"});
}
std::string GetLoadableModuleName(std::string_view name) {
return GetNativeLibraryName(name);
}
} // namespace base