1*6777b538SAndroid Build Coastguard Worker // Copyright 2019 The Chromium Authors
2*6777b538SAndroid Build Coastguard Worker // Use of this source code is governed by a BSD-style license that can be
3*6777b538SAndroid Build Coastguard Worker // found in the LICENSE file.
4*6777b538SAndroid Build Coastguard Worker
5*6777b538SAndroid Build Coastguard Worker #include "base/process/environment_internal.h"
6*6777b538SAndroid Build Coastguard Worker
7*6777b538SAndroid Build Coastguard Worker #include <stddef.h>
8*6777b538SAndroid Build Coastguard Worker
9*6777b538SAndroid Build Coastguard Worker #include <vector>
10*6777b538SAndroid Build Coastguard Worker
11*6777b538SAndroid Build Coastguard Worker #include "build/build_config.h"
12*6777b538SAndroid Build Coastguard Worker
13*6777b538SAndroid Build Coastguard Worker #if BUILDFLAG(IS_POSIX) || BUILDFLAG(IS_FUCHSIA)
14*6777b538SAndroid Build Coastguard Worker #include <string.h>
15*6777b538SAndroid Build Coastguard Worker #endif
16*6777b538SAndroid Build Coastguard Worker
17*6777b538SAndroid Build Coastguard Worker #if BUILDFLAG(IS_WIN)
18*6777b538SAndroid Build Coastguard Worker #include "base/check_op.h"
19*6777b538SAndroid Build Coastguard Worker #endif
20*6777b538SAndroid Build Coastguard Worker
21*6777b538SAndroid Build Coastguard Worker namespace base {
22*6777b538SAndroid Build Coastguard Worker namespace internal {
23*6777b538SAndroid Build Coastguard Worker
24*6777b538SAndroid Build Coastguard Worker namespace {
25*6777b538SAndroid Build Coastguard Worker
26*6777b538SAndroid Build Coastguard Worker #if BUILDFLAG(IS_POSIX) || BUILDFLAG(IS_FUCHSIA) || BUILDFLAG(IS_WIN)
27*6777b538SAndroid Build Coastguard Worker // Parses a null-terminated input string of an environment block. The key is
28*6777b538SAndroid Build Coastguard Worker // placed into the given string, and the total length of the line, including
29*6777b538SAndroid Build Coastguard Worker // the terminating null, is returned.
ParseEnvLine(const NativeEnvironmentString::value_type * input,NativeEnvironmentString * key)30*6777b538SAndroid Build Coastguard Worker size_t ParseEnvLine(const NativeEnvironmentString::value_type* input,
31*6777b538SAndroid Build Coastguard Worker NativeEnvironmentString* key) {
32*6777b538SAndroid Build Coastguard Worker // Skip to the equals or end of the string, this is the key.
33*6777b538SAndroid Build Coastguard Worker size_t cur = 0;
34*6777b538SAndroid Build Coastguard Worker while (input[cur] && input[cur] != '=')
35*6777b538SAndroid Build Coastguard Worker cur++;
36*6777b538SAndroid Build Coastguard Worker *key = NativeEnvironmentString(&input[0], cur);
37*6777b538SAndroid Build Coastguard Worker
38*6777b538SAndroid Build Coastguard Worker // Now just skip to the end of the string.
39*6777b538SAndroid Build Coastguard Worker while (input[cur])
40*6777b538SAndroid Build Coastguard Worker cur++;
41*6777b538SAndroid Build Coastguard Worker return cur + 1;
42*6777b538SAndroid Build Coastguard Worker }
43*6777b538SAndroid Build Coastguard Worker #endif
44*6777b538SAndroid Build Coastguard Worker
45*6777b538SAndroid Build Coastguard Worker } // namespace
46*6777b538SAndroid Build Coastguard Worker
47*6777b538SAndroid Build Coastguard Worker #if BUILDFLAG(IS_POSIX) || BUILDFLAG(IS_FUCHSIA)
48*6777b538SAndroid Build Coastguard Worker
AlterEnvironment(const char * const * const env,const EnvironmentMap & changes)49*6777b538SAndroid Build Coastguard Worker std::unique_ptr<char* []> AlterEnvironment(const char* const* const env,
50*6777b538SAndroid Build Coastguard Worker const EnvironmentMap& changes) {
51*6777b538SAndroid Build Coastguard Worker std::string value_storage; // Holds concatenated null-terminated strings.
52*6777b538SAndroid Build Coastguard Worker std::vector<size_t> result_indices; // Line indices into value_storage.
53*6777b538SAndroid Build Coastguard Worker
54*6777b538SAndroid Build Coastguard Worker // First build up all of the unchanged environment strings. These are
55*6777b538SAndroid Build Coastguard Worker // null-terminated of the form "key=value".
56*6777b538SAndroid Build Coastguard Worker std::string key;
57*6777b538SAndroid Build Coastguard Worker for (size_t i = 0; env[i]; i++) {
58*6777b538SAndroid Build Coastguard Worker size_t line_length = ParseEnvLine(env[i], &key);
59*6777b538SAndroid Build Coastguard Worker
60*6777b538SAndroid Build Coastguard Worker // Keep only values not specified in the change vector.
61*6777b538SAndroid Build Coastguard Worker auto found_change = changes.find(key);
62*6777b538SAndroid Build Coastguard Worker if (found_change == changes.end()) {
63*6777b538SAndroid Build Coastguard Worker result_indices.push_back(value_storage.size());
64*6777b538SAndroid Build Coastguard Worker value_storage.append(env[i], line_length);
65*6777b538SAndroid Build Coastguard Worker }
66*6777b538SAndroid Build Coastguard Worker }
67*6777b538SAndroid Build Coastguard Worker
68*6777b538SAndroid Build Coastguard Worker // Now append all modified and new values.
69*6777b538SAndroid Build Coastguard Worker for (const auto& i : changes) {
70*6777b538SAndroid Build Coastguard Worker if (!i.second.empty()) {
71*6777b538SAndroid Build Coastguard Worker result_indices.push_back(value_storage.size());
72*6777b538SAndroid Build Coastguard Worker value_storage.append(i.first);
73*6777b538SAndroid Build Coastguard Worker value_storage.push_back('=');
74*6777b538SAndroid Build Coastguard Worker value_storage.append(i.second);
75*6777b538SAndroid Build Coastguard Worker value_storage.push_back(0);
76*6777b538SAndroid Build Coastguard Worker }
77*6777b538SAndroid Build Coastguard Worker }
78*6777b538SAndroid Build Coastguard Worker
79*6777b538SAndroid Build Coastguard Worker size_t pointer_count_required =
80*6777b538SAndroid Build Coastguard Worker result_indices.size() + 1 + // Null-terminated array of pointers.
81*6777b538SAndroid Build Coastguard Worker (value_storage.size() + sizeof(char*) - 1) / sizeof(char*); // Buffer.
82*6777b538SAndroid Build Coastguard Worker std::unique_ptr<char*[]> result(new char*[pointer_count_required]);
83*6777b538SAndroid Build Coastguard Worker
84*6777b538SAndroid Build Coastguard Worker // The string storage goes after the array of pointers.
85*6777b538SAndroid Build Coastguard Worker char* storage_data =
86*6777b538SAndroid Build Coastguard Worker reinterpret_cast<char*>(&result.get()[result_indices.size() + 1]);
87*6777b538SAndroid Build Coastguard Worker if (!value_storage.empty())
88*6777b538SAndroid Build Coastguard Worker memcpy(storage_data, value_storage.data(), value_storage.size());
89*6777b538SAndroid Build Coastguard Worker
90*6777b538SAndroid Build Coastguard Worker // Fill array of pointers at the beginning of the result.
91*6777b538SAndroid Build Coastguard Worker for (size_t i = 0; i < result_indices.size(); i++)
92*6777b538SAndroid Build Coastguard Worker result[i] = &storage_data[result_indices[i]];
93*6777b538SAndroid Build Coastguard Worker result[result_indices.size()] = 0; // Null terminator.
94*6777b538SAndroid Build Coastguard Worker
95*6777b538SAndroid Build Coastguard Worker return result;
96*6777b538SAndroid Build Coastguard Worker }
97*6777b538SAndroid Build Coastguard Worker
98*6777b538SAndroid Build Coastguard Worker #elif BUILDFLAG(IS_WIN)
99*6777b538SAndroid Build Coastguard Worker
AlterEnvironment(const wchar_t * env,const EnvironmentMap & changes)100*6777b538SAndroid Build Coastguard Worker NativeEnvironmentString AlterEnvironment(const wchar_t* env,
101*6777b538SAndroid Build Coastguard Worker const EnvironmentMap& changes) {
102*6777b538SAndroid Build Coastguard Worker NativeEnvironmentString result;
103*6777b538SAndroid Build Coastguard Worker
104*6777b538SAndroid Build Coastguard Worker // First build up all of the unchanged environment strings.
105*6777b538SAndroid Build Coastguard Worker const wchar_t* ptr = env;
106*6777b538SAndroid Build Coastguard Worker while (*ptr) {
107*6777b538SAndroid Build Coastguard Worker std::wstring key;
108*6777b538SAndroid Build Coastguard Worker size_t line_length = ParseEnvLine(ptr, &key);
109*6777b538SAndroid Build Coastguard Worker
110*6777b538SAndroid Build Coastguard Worker // Keep only values not specified in the change vector.
111*6777b538SAndroid Build Coastguard Worker if (changes.find(key) == changes.end()) {
112*6777b538SAndroid Build Coastguard Worker result.append(ptr, line_length);
113*6777b538SAndroid Build Coastguard Worker }
114*6777b538SAndroid Build Coastguard Worker ptr += line_length;
115*6777b538SAndroid Build Coastguard Worker }
116*6777b538SAndroid Build Coastguard Worker
117*6777b538SAndroid Build Coastguard Worker // Now append all modified and new values.
118*6777b538SAndroid Build Coastguard Worker for (const auto& i : changes) {
119*6777b538SAndroid Build Coastguard Worker // Windows environment blocks cannot handle keys or values with NULs.
120*6777b538SAndroid Build Coastguard Worker CHECK_EQ(std::wstring::npos, i.first.find(L'\0'));
121*6777b538SAndroid Build Coastguard Worker CHECK_EQ(std::wstring::npos, i.second.find(L'\0'));
122*6777b538SAndroid Build Coastguard Worker if (!i.second.empty()) {
123*6777b538SAndroid Build Coastguard Worker result += i.first;
124*6777b538SAndroid Build Coastguard Worker result.push_back('=');
125*6777b538SAndroid Build Coastguard Worker result += i.second;
126*6777b538SAndroid Build Coastguard Worker result.push_back('\0');
127*6777b538SAndroid Build Coastguard Worker }
128*6777b538SAndroid Build Coastguard Worker }
129*6777b538SAndroid Build Coastguard Worker
130*6777b538SAndroid Build Coastguard Worker // Add the terminating NUL.
131*6777b538SAndroid Build Coastguard Worker result.push_back('\0');
132*6777b538SAndroid Build Coastguard Worker return result;
133*6777b538SAndroid Build Coastguard Worker }
134*6777b538SAndroid Build Coastguard Worker
135*6777b538SAndroid Build Coastguard Worker #endif // BUILDFLAG(IS_POSIX) || BUILDFLAG(IS_FUCHSIA)
136*6777b538SAndroid Build Coastguard Worker
137*6777b538SAndroid Build Coastguard Worker } // namespace internal
138*6777b538SAndroid Build Coastguard Worker } // namespace base
139