2
2
// Licensed under the MIT License
3
3
4
4
#include " core/providers/qnn/rpcmem_library.h"
5
+
6
+ #if defined(_WIN32)
7
+ #include < filesystem>
8
+
9
+ #include < sysinfoapi.h>
10
+ #include < winsvc.h>
11
+ #endif // defined(_WIN32)
12
+
5
13
#include " core/providers/qnn/ort_api.h"
6
14
7
15
namespace onnxruntime ::qnn {
8
16
17
+ // Unload the dynamic library referenced by `library_handle`.
18
+ // Avoid throwing because this may run from a dtor.
19
+ void DynamicLibraryHandleDeleter::operator ()(void * library_handle) noexcept {
20
+ if (library_handle == nullptr ) {
21
+ return ;
22
+ }
23
+
24
+ const auto & env = GetDefaultEnv ();
25
+ const auto unload_status = env.UnloadDynamicLibrary (library_handle);
26
+
27
+ if (!unload_status.IsOK ()) {
28
+ LOGS_DEFAULT (WARNING) << " Failed to unload dynamic library. Error: " << unload_status.ErrorMessage ();
29
+ }
30
+ }
31
+
9
32
namespace {
10
33
11
- const PathChar* GetRpcMemSharedLibraryPath () {
12
34
#if defined(_WIN32)
13
- return ORT_TSTR (" libcdsprpc.dll" );
14
- #else
15
- return ORT_TSTR (" libcdsprpc.so" );
16
- #endif
35
+
36
+ struct ServiceHandleDeleter {
37
+ void operator ()(SC_HANDLE handle) { ::CloseServiceHandle (handle); }
38
+ };
39
+
40
+ using UniqueServiceHandle = std::unique_ptr<std::remove_pointer_t <SC_HANDLE>, ServiceHandleDeleter>;
41
+
42
+ Status ReadEnvironmentVariable (const wchar_t * name, std::wstring& value_out) {
43
+ const DWORD value_size = ::GetEnvironmentVariableW (name, nullptr , 0 );
44
+ ORT_RETURN_IF (value_size == 0 ,
45
+ " Failed to get environment variable length. GetEnvironmentVariableW error: " , ::GetLastError ());
46
+
47
+ std::vector<wchar_t > value (value_size);
48
+
49
+ ORT_RETURN_IF (::GetEnvironmentVariableW (name, value.data (), value_size) == 0 ,
50
+ " Failed to get environment variable value. GetEnvironmentVariableW error: " , ::GetLastError ());
51
+
52
+ value_out = std::wstring{value.data ()};
53
+ return Status::OK ();
17
54
}
18
55
19
- DynamicLibraryHandle LoadDynamicLibrary (const PathString& path, bool global_symbols) {
20
- // Custom deleter to unload the shared library. Avoid throwing from it because it may run in dtor.
21
- const auto unload_library = [](void * library_handle) {
22
- if (library_handle == nullptr ) {
23
- return ;
24
- }
56
+ Status GetServiceBinaryDirectoryPath (const wchar_t * service_name,
57
+ std::filesystem::path& service_binary_directory_path_out) {
58
+ SC_HANDLE scm_handle_raw = ::OpenSCManagerW (nullptr , // local computer
59
+ nullptr , // SERVICES_ACTIVE_DATABASE
60
+ STANDARD_RIGHTS_READ);
61
+ ORT_RETURN_IF (scm_handle_raw == nullptr ,
62
+ " Failed to open handle to service control manager. OpenSCManagerW error: " , ::GetLastError ());
63
+
64
+ auto scm_handle = UniqueServiceHandle{scm_handle_raw};
65
+
66
+ SC_HANDLE service_handle_raw = ::OpenServiceW (scm_handle.get (),
67
+ service_name,
68
+ SERVICE_QUERY_CONFIG);
69
+ ORT_RETURN_IF (service_handle_raw == nullptr ,
70
+ " Failed to open service handle. OpenServiceW error: " , ::GetLastError ());
71
+
72
+ auto service_handle = UniqueServiceHandle{service_handle_raw};
73
+
74
+ // get service config required buffer size
75
+ DWORD service_config_buffer_size{};
76
+ ORT_RETURN_IF (!::QueryServiceConfigW (service_handle.get (), nullptr , 0 , &service_config_buffer_size) &&
77
+ ::GetLastError () != ERROR_INSUFFICIENT_BUFFER,
78
+ "Failed to query service configuration buffer size. QueryServiceConfigW error: ", ::GetLastError());
25
79
26
- const auto & env = GetDefaultEnv ();
27
- const auto unload_status = env.UnloadDynamicLibrary (library_handle);
80
+ // get the service config
81
+ std::vector<std::byte> service_config_buffer (service_config_buffer_size);
82
+ QUERY_SERVICE_CONFIGW* service_config = reinterpret_cast <QUERY_SERVICE_CONFIGW*>(service_config_buffer.data ());
83
+ ORT_RETURN_IF (!::QueryServiceConfigW (service_handle.get (), service_config, service_config_buffer_size,
84
+ &service_config_buffer_size),
85
+ " Failed to query service configuration. QueryServiceConfigW error: " , ::GetLastError ());
28
86
29
- if (!unload_status.IsOK ()) {
30
- LOGS_DEFAULT (WARNING) << " Failed to unload shared library. Error: " << unload_status.ErrorMessage ();
31
- }
32
- };
87
+ std::wstring service_binary_path_name = service_config->lpBinaryPathName ;
33
88
89
+ // replace system root placeholder with the value of the SYSTEMROOT environment variable
90
+ const std::wstring system_root_placeholder = L" \\ SystemRoot" ;
91
+
92
+ ORT_RETURN_IF (service_binary_path_name.find (system_root_placeholder, 0 ) != 0 ,
93
+ " Service binary path '" , ToUTF8String (service_binary_path_name),
94
+ " ' does not start with expected system root placeholder value '" ,
95
+ ToUTF8String (system_root_placeholder), " '." );
96
+
97
+ std::wstring system_root{};
98
+ ORT_RETURN_IF_ERROR (ReadEnvironmentVariable (L" SYSTEMROOT" , system_root));
99
+ service_binary_path_name.replace (0 , system_root_placeholder.size (), system_root);
100
+
101
+ const auto service_binary_path = std::filesystem::path{service_binary_path_name};
102
+ auto service_binary_directory_path = service_binary_path.parent_path ();
103
+
104
+ ORT_RETURN_IF (!std::filesystem::exists (service_binary_directory_path),
105
+ " Service binary directory path does not exist: " , service_binary_directory_path.string ());
106
+
107
+ service_binary_directory_path_out = std::move (service_binary_directory_path);
108
+ return Status::OK ();
109
+ }
110
+
111
+ #endif // defined(_WIN32)
112
+
113
+ Status GetRpcMemDynamicLibraryPath (PathString& path_out) {
114
+ #if defined(_WIN32)
115
+
116
+ std::filesystem::path qcnspmcdm_dir_path{};
117
+ ORT_RETURN_IF_ERROR (GetServiceBinaryDirectoryPath (L" qcnspmcdm" , qcnspmcdm_dir_path));
118
+ const auto libcdsprpc_path = qcnspmcdm_dir_path / L" libcdsprpc.dll" ;
119
+ path_out = libcdsprpc_path.wstring ();
120
+ return Status::OK ();
121
+
122
+ #else // ^^^ defined(_WIN32) / vvv !defined(_WIN32)
123
+
124
+ path_out = ORT_TSTR (" libcdsprpc.so" );
125
+ return Status::OK ();
126
+
127
+ #endif // !defined(_WIN32)
128
+ }
129
+
130
+ Status LoadDynamicLibrary (const PathString& path, bool global_symbols,
131
+ UniqueDynamicLibraryHandle& library_handle_out) {
34
132
const auto & env = GetDefaultEnv ();
35
- void * library_handle = nullptr ;
133
+ void * library_handle_raw = nullptr ;
134
+ ORT_RETURN_IF_ERROR (env.LoadDynamicLibrary (path, global_symbols, &library_handle_raw));
135
+
136
+ library_handle_out = UniqueDynamicLibraryHandle{library_handle_raw};
137
+ return Status::OK ();
138
+ }
139
+
140
+ UniqueDynamicLibraryHandle GetRpcMemDynamicLibraryHandle () {
141
+ std::string_view error_message_prefix = " Failed to initialize RPCMEM dynamic library handle: " ;
142
+
143
+ PathString rpcmem_library_path{};
144
+ auto status = GetRpcMemDynamicLibraryPath (rpcmem_library_path);
145
+ if (!status.IsOK ()) {
146
+ ORT_THROW (error_message_prefix, status.ErrorMessage ());
147
+ }
36
148
37
- const auto load_status = env.LoadDynamicLibrary (path, global_symbols, &library_handle);
38
- if (!load_status.IsOK ()) {
39
- ORT_THROW (" Failed to load " , ToUTF8String (path), " : " , load_status.ErrorMessage ());
149
+ UniqueDynamicLibraryHandle library_handle{};
150
+ status = LoadDynamicLibrary (rpcmem_library_path, /* global_symbols */ false , library_handle);
151
+ if (!status.IsOK ()) {
152
+ ORT_THROW (error_message_prefix, status.ErrorMessage ());
40
153
}
41
154
42
- return DynamicLibraryHandle{ library_handle, unload_library} ;
155
+ return library_handle;
43
156
}
44
157
45
158
RpcMemApi CreateApi (void * library_handle) {
@@ -58,7 +171,7 @@ RpcMemApi CreateApi(void* library_handle) {
58
171
} // namespace
59
172
60
173
RpcMemLibrary::RpcMemLibrary ()
61
- : library_handle_(LoadDynamicLibrary(GetRpcMemSharedLibraryPath(), /* global_symbols */ false )),
174
+ : library_handle_(GetRpcMemDynamicLibraryHandle( )),
62
175
api_{CreateApi (library_handle_.get ())} {
63
176
}
64
177
0 commit comments