ccls/src/platform_posix.cc

173 lines
4.5 KiB
C++
Raw Normal View History

2018-08-21 05:27:52 +00:00
/* Copyright 2017-2018 ccls Authors
Licensed under the Apache License, Version 2.0 (the "License");
you may not use this file except in compliance with the License.
You may obtain a copy of the License at
http://www.apache.org/licenses/LICENSE-2.0
Unless required by applicable law or agreed to in writing, software
distributed under the License is distributed on an "AS IS" BASIS,
WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
See the License for the specific language governing permissions and
limitations under the License.
==============================================================================*/
#if defined(__unix__) || defined(__APPLE__)
2017-03-04 01:45:20 +00:00
#include "platform.h"
2017-03-25 20:15:00 +00:00
#include "utils.h"
2017-03-04 01:45:20 +00:00
#include <assert.h>
2017-09-22 01:14:57 +00:00
#include <limits.h>
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <time.h>
2017-04-18 17:21:53 +00:00
#include <dirent.h>
#include <errno.h>
#include <fcntl.h>
#include <signal.h>
#include <sys/stat.h>
2018-08-09 17:08:14 +00:00
#include <sys/types.h> // required for stat.h
#include <sys/wait.h>
2018-08-09 17:08:14 +00:00
#include <unistd.h>
#ifdef __GLIBC__
2017-12-23 16:01:43 +00:00
#include <malloc.h>
2017-10-31 22:43:07 +00:00
#endif
2017-12-29 22:46:21 +00:00
#include <string>
namespace {
2017-12-01 17:50:39 +00:00
// Returns the canonicalized absolute pathname, without expanding symbolic
// links. This is a variant of realpath(2), C++ rewrite of
// https://github.com/freebsd/freebsd/blob/master/lib/libc/stdlib/realpath.c
2018-03-31 03:16:33 +00:00
std::optional<std::string> RealPathNotExpandSymlink(std::string path) {
if (path.empty()) {
errno = EINVAL;
2018-03-31 03:16:33 +00:00
return std::nullopt;
}
if (path[0] == '\0') {
errno = ENOENT;
2018-03-31 03:16:33 +00:00
return std::nullopt;
}
// Do not use PATH_MAX because it is tricky on Linux.
// See https://eklitzke.org/path-max-is-tricky
char tmp[1024];
std::string resolved;
size_t i = 0;
struct stat sb;
if (path[0] == '/') {
resolved = "/";
i = 1;
} else {
if (!getcwd(tmp, sizeof tmp))
2018-03-31 03:16:33 +00:00
return std::nullopt;
resolved = tmp;
}
while (i < path.size()) {
auto j = path.find('/', i);
if (j == std::string::npos)
j = path.size();
auto next_token = path.substr(i, j - i);
i = j + 1;
if (resolved.back() != '/')
resolved += '/';
if (next_token.empty() || next_token == ".") {
// Handle consequential slashes and "."
continue;
} else if (next_token == "..") {
// Strip the last path component except when it is single "/"
if (resolved.size() > 1)
resolved.resize(resolved.rfind('/', resolved.size() - 2) + 1);
continue;
}
// Append the next path component.
// Here we differ from realpath(3), we use stat(2) instead of
// lstat(2) because we do not want to resolve symlinks.
resolved += next_token;
if (stat(resolved.c_str(), &sb) != 0)
2018-03-31 03:16:33 +00:00
return std::nullopt;
if (!S_ISDIR(sb.st_mode) && j < path.size()) {
errno = ENOTDIR;
2018-03-31 03:16:33 +00:00
return std::nullopt;
}
}
// Remove trailing slash except when a single "/".
if (resolved.size() > 1 && resolved.back() == '/')
resolved.pop_back();
return resolved;
}
2018-08-09 17:08:14 +00:00
} // namespace
2018-08-09 17:08:14 +00:00
std::string NormalizePath(const std::string &path) {
2018-03-31 03:16:33 +00:00
std::optional<std::string> resolved = RealPathNotExpandSymlink(path);
return resolved ? *resolved : path;
2017-03-28 01:47:12 +00:00
}
2017-08-17 18:02:47 +00:00
void FreeUnusedMemory() {
#ifdef __GLIBC__
malloc_trim(4 * 1024 * 1024);
2017-10-31 22:43:07 +00:00
#endif
2017-08-17 18:02:47 +00:00
}
void TraceMe() {
// If the environment variable is defined, wait for a debugger.
2018-03-31 03:16:33 +00:00
// In gdb, you need to invoke `signal SIGCONT` if you want ccls to continue
// after detaching.
2018-08-09 17:08:14 +00:00
const char *traceme = getenv("CCLS_TRACEME");
if (traceme)
raise(traceme[0] == 's' ? SIGSTOP : SIGTSTP);
}
2018-08-09 17:08:14 +00:00
std::string GetExternalCommandOutput(const std::vector<std::string> &command,
std::string_view input) {
int pin[2], pout[2];
2018-04-02 03:55:10 +00:00
if (pipe(pin) < 0) {
perror("pipe(stdin)");
2018-02-20 01:19:50 +00:00
return "";
2018-04-02 03:55:10 +00:00
}
2018-02-20 01:19:50 +00:00
if (pipe(pout) < 0) {
2018-04-02 03:55:10 +00:00
perror("pipe(stdout)");
2018-02-20 01:19:50 +00:00
close(pin[0]);
close(pin[1]);
return "";
}
pid_t child = fork();
if (child == 0) {
dup2(pout[0], 0);
dup2(pin[1], 1);
close(pin[0]);
close(pin[1]);
close(pout[0]);
close(pout[1]);
2018-08-09 17:08:14 +00:00
auto argv = new char *[command.size() + 1];
for (size_t i = 0; i < command.size(); i++)
2018-08-09 17:08:14 +00:00
argv[i] = const_cast<char *>(command[i].c_str());
argv[command.size()] = nullptr;
execvp(argv[0], argv);
_Exit(127);
}
close(pin[1]);
close(pout[0]);
2018-02-20 03:06:48 +00:00
// O_NONBLOCK is disabled, write(2) blocks until all bytes are written.
(void)write(pout[1], input.data(), input.size());
close(pout[1]);
std::string ret;
char buf[4096];
ssize_t n;
while ((n = read(pin[0], buf, sizeof buf)) > 0)
ret.append(buf, n);
2018-04-02 03:55:10 +00:00
close(pin[0]);
waitpid(child, NULL, 0);
return ret;
}
2017-03-21 17:06:22 +00:00
#endif