2020-05-02 10:21:10 -07:00
|
|
|
// This file is part of river, a dynamic tiling wayland compositor.
|
|
|
|
//
|
2020-11-11 11:30:21 -08:00
|
|
|
// Copyright 2020 The River Developers
|
2020-05-02 10:21:10 -07:00
|
|
|
//
|
|
|
|
// This program is free software: you can redistribute it and/or modify
|
|
|
|
// it under the terms of the GNU General Public License as published by
|
|
|
|
// the Free Software Foundation, either version 3 of the License, or
|
|
|
|
// (at your option) any later version.
|
|
|
|
//
|
|
|
|
// This program is distributed in the hope that it will be useful,
|
|
|
|
// but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
|
|
// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
|
|
// GNU General Public License for more details.
|
|
|
|
//
|
|
|
|
// You should have received a copy of the GNU General Public License
|
|
|
|
// along with this program. If not, see <https://www.gnu.org/licenses/>.
|
|
|
|
|
2020-04-18 04:41:04 -07:00
|
|
|
const std = @import("std");
|
|
|
|
|
2020-07-08 05:00:51 -07:00
|
|
|
const c = @import("../c.zig");
|
|
|
|
|
2020-05-26 13:55:07 -07:00
|
|
|
const Error = @import("../command.zig").Error;
|
2020-05-02 14:11:56 -07:00
|
|
|
const Seat = @import("../Seat.zig");
|
2020-04-18 04:41:04 -07:00
|
|
|
|
|
|
|
/// Spawn a program.
|
2020-05-26 13:55:07 -07:00
|
|
|
pub fn spawn(
|
|
|
|
allocator: *std.mem.Allocator,
|
|
|
|
seat: *Seat,
|
|
|
|
args: []const []const u8,
|
2020-06-26 08:57:03 -07:00
|
|
|
out: *?[]const u8,
|
2020-05-26 13:55:07 -07:00
|
|
|
) Error!void {
|
|
|
|
if (args.len < 2) return Error.NotEnoughArguments;
|
|
|
|
|
|
|
|
const cmd = try std.mem.join(allocator, " ", args[1..]);
|
|
|
|
defer allocator.free(cmd);
|
2020-07-08 05:00:51 -07:00
|
|
|
const cmdZ = try std.cstr.addNullByte(allocator, cmd);
|
|
|
|
defer allocator.free(cmdZ);
|
2020-05-26 13:55:07 -07:00
|
|
|
|
2020-07-08 05:00:51 -07:00
|
|
|
const child_args = [_:null]?[*:0]const u8{ "/bin/sh", "-c", cmdZ, null };
|
2020-04-18 04:41:04 -07:00
|
|
|
|
2020-07-08 05:00:51 -07:00
|
|
|
const pid = std.os.fork() catch {
|
|
|
|
out.* = try std.fmt.allocPrint(allocator, "fork/execve failed", .{});
|
2020-06-26 08:57:03 -07:00
|
|
|
return Error.Other;
|
2020-04-18 04:41:04 -07:00
|
|
|
};
|
2020-07-08 05:00:51 -07:00
|
|
|
|
|
|
|
if (pid == 0) {
|
|
|
|
// Clean things up for the child in an intermediate fork
|
|
|
|
if (c.setsid() < 0) unreachable;
|
|
|
|
if (std.os.system.sigprocmask(std.os.SIG_SETMASK, &std.os.empty_sigset, null) < 0) unreachable;
|
|
|
|
|
2020-07-12 03:16:05 -07:00
|
|
|
const pid2 = std.os.fork() catch c._exit(1);
|
|
|
|
if (pid2 == 0) std.os.execveZ("/bin/sh", &child_args, std.c.environ) catch c._exit(1);
|
2020-07-08 05:00:51 -07:00
|
|
|
|
2020-07-12 03:16:05 -07:00
|
|
|
c._exit(0);
|
2020-07-08 05:00:51 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
// Wait the intermediate child.
|
2020-10-22 09:20:09 -07:00
|
|
|
const ret = std.os.waitpid(pid, 0);
|
|
|
|
if (!std.os.WIFEXITED(ret.status) or
|
|
|
|
(std.os.WIFEXITED(ret.status) and std.os.WEXITSTATUS(ret.status) != 0))
|
2020-07-12 03:16:05 -07:00
|
|
|
{
|
2020-07-08 05:00:51 -07:00
|
|
|
out.* = try std.fmt.allocPrint(allocator, "fork/execve failed", .{});
|
|
|
|
return Error.Other;
|
|
|
|
}
|
2020-04-18 04:41:04 -07:00
|
|
|
}
|