nvim_get_proc_children: fallback to shell

/proc/…/children may be unavailable because of an unset kernel option.
Fallback to `pgrep` invoked in a shell.
This commit is contained in:
Justin M. Keyes 2018-03-14 23:26:37 +01:00
parent dbad797edd
commit 12af7016e2
3 changed files with 43 additions and 8 deletions

View File

@ -1482,7 +1482,7 @@ Array nvim_list_uis(void)
/// Gets the immediate children of process `pid`. /// Gets the immediate children of process `pid`.
/// ///
/// @return Array of child process ids, or empty array if process not found. /// @return Array of child process ids. Empty array if process not found.
Array nvim_get_proc_children(Integer pid, Error *err) Array nvim_get_proc_children(Integer pid, Error *err)
FUNC_API_SINCE(4) FUNC_API_SINCE(4)
{ {
@ -1490,17 +1490,27 @@ Array nvim_get_proc_children(Integer pid, Error *err)
int *proc_list = NULL; int *proc_list = NULL;
if (pid <= 0 || pid > INT_MAX) { if (pid <= 0 || pid > INT_MAX) {
api_set_error(err, kErrorTypeException, "Invalid pid: %d", pid); api_set_error(err, kErrorTypeException, "Invalid pid: %" PRId64, pid);
goto end; goto end;
} }
size_t proc_count; size_t proc_count;
int rv = os_proc_children((int)pid, &proc_list, &proc_count); int rv = os_proc_children((int)pid, &proc_list, &proc_count);
if (rv == 1) { if (rv != 0) {
goto end; // Process not found; return empty list. // syscall failed (possibly because of kernel options), try shelling out.
} else if (rv != 0) { Array a = ARRAY_DICT_INIT;
api_set_error(err, kErrorTypeException, ADD(a, INTEGER_OBJ(pid));
"Failed to get process children. pid=%d error=%d", pid, rv); String s = cstr_to_string("return vim._os_proc_children(select(1, ...))");
Object o = nvim_execute_lua(s, a, err);
api_free_string(s);
api_free_array(a);
if (o.type == kObjectTypeArray) {
proc_array = o.data.array;
} else if (!ERROR_SET(err)) {
api_set_error(err, kErrorTypeException,
"Failed to get process children. pid=%" PRId64 " error=%d",
pid, rv);
}
goto end; goto end;
} }

View File

@ -1,3 +1,26 @@
-- Gets the children of process `ppid` via the shell.
-- Used by nvim_get_proc_children() as a fallback.
local function _os_proc_children(ppid)
if ppid == nil or ppid <= 0 or type(ppid) ~= 'number' then
error('invalid ppid')
end
local out = vim.api.nvim_call_function('system', { 'pgrep -P '..ppid })
local err = vim.api.nvim_get_vvar('shell_error')
if 1 == err and out == '' then
return {} -- Process not found.
elseif 0 ~= err then
error('pgrep failed')
end
local children = {}
for s in string.gmatch(out, '%S+') do
local i = tonumber(s)
if i ~= nil then
table.insert(children, i)
end
end
return children
end
-- TODO(ZyX-I): Create compatibility layer. -- TODO(ZyX-I): Create compatibility layer.
--{{{1 package.path updater function --{{{1 package.path updater function
-- Last inserted paths. Used to clear out items from package.[c]path when they -- Last inserted paths. Used to clear out items from package.[c]path when they
@ -61,4 +84,5 @@ end
--{{{1 Module definition --{{{1 Module definition
return { return {
_update_package_paths = _update_package_paths, _update_package_paths = _update_package_paths,
_os_proc_children = _os_proc_children,
} }

View File

@ -2,6 +2,7 @@
// it. PVS-Studio Static Code Analyzer for C, C++ and C#: http://www.viva64.com // it. PVS-Studio Static Code Analyzer for C, C++ and C#: http://www.viva64.com
#include <uv.h> // for HANDLE (win32) #include <uv.h> // for HANDLE (win32)
#ifdef WIN32 #ifdef WIN32
# include <tlhelp32.h> // for CreateToolhelp32Snapshot # include <tlhelp32.h> // for CreateToolhelp32Snapshot
#endif #endif
@ -165,7 +166,7 @@ int os_proc_children(int ppid, int **proc_list, size_t *proc_count)
snprintf(proc_p, sizeof(proc_p), "/proc/%d/task/%d/children", ppid, ppid); snprintf(proc_p, sizeof(proc_p), "/proc/%d/task/%d/children", ppid, ppid);
FILE *fp = fopen(proc_p, "r"); FILE *fp = fopen(proc_p, "r");
if (fp == NULL) { if (fp == NULL) {
return 1; // Process not found. return 2; // Process not found, or /proc/…/children not supported.
} }
int match_pid; int match_pid;
while (fscanf(fp, "%d", &match_pid) > 0) { while (fscanf(fp, "%d", &match_pid) > 0) {