aboutsummaryrefslogtreecommitdiffstats
path: root/nihil.posix/getenv.cc
blob: c5969028d5ed9501340078f4b1f6150b7f09255b (plain) (blame)
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
/*
 * This source code is released into the public domain.
 */

module;

#include <cstdint>
#include <expected>
#include <string>
#include <system_error>
#include <vector>

#include <unistd.h>

#include "nihil.hh"

module nihil.posix;

import nihil.error;

namespace nihil {

auto getenv(std::string_view varname) -> std::expected<std::string, error>
{
	auto cvarname = std::string(varname);

#ifdef NIHIL_HAVE_GETENV_R
	// Start with a buffer of this size, and double it every iteration.
	constexpr auto bufinc = std::size_t{1024};

	auto buf = std::vector<char>(bufinc);
	for (;;) {
		auto const ret = ::getenv_r(cvarname.c_str(),
					    buf.data(), buf.size());

		if (ret == 0)
			return {std::string(buf.data())};

		if (ret == -1 && errno == ERANGE) {
			buf.resize(buf.size() * 2);
			continue;
		}

		return std::unexpected(error(std::errc(errno)));
	}
#else // NIHIL_HAVE_GETENV_R
	errno = 0;
	auto *v = ::getenv(cvarname.c_str());

	if (v != nullptr)
		return {std::string(v)};

	if (errno != 0)
		return std::unexpected(error(std::errc(errno)));

	return std::unexpected(error(std::errc::no_such_file_or_directory));
#endif // NIHIL_HAVE_GETENV_R
}

} // namespace nihil