1 //===-- Program.cpp - Implement OS Program Concept --------------*- C++ -*-===//
3 // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
4 // See https://llvm.org/LICENSE.txt for license information.
5 // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
7 //===----------------------------------------------------------------------===//
9 // This file implements the operating system Program concept.
11 //===----------------------------------------------------------------------===//
13 #include "llvm/Support/Program.h"
14 #include "llvm/ADT/StringRef.h"
15 #include "llvm/Config/llvm-config.h"
16 #include "llvm/Support/raw_ostream.h"
20 //===----------------------------------------------------------------------===//
21 //=== WARNING: Implementation here must contain only TRULY operating system
22 //=== independent code.
23 //===----------------------------------------------------------------------===//
25 static bool Execute(ProcessInfo
&PI
, StringRef Program
,
26 ArrayRef
<StringRef
> Args
,
27 std::optional
<ArrayRef
<StringRef
>> Env
,
28 ArrayRef
<std::optional
<StringRef
>> Redirects
,
29 unsigned MemoryLimit
, std::string
*ErrMsg
,
30 BitVector
*AffinityMask
, bool DetachProcess
);
32 int sys::ExecuteAndWait(StringRef Program
, ArrayRef
<StringRef
> Args
,
33 std::optional
<ArrayRef
<StringRef
>> Env
,
34 ArrayRef
<std::optional
<StringRef
>> Redirects
,
35 unsigned SecondsToWait
, unsigned MemoryLimit
,
36 std::string
*ErrMsg
, bool *ExecutionFailed
,
37 std::optional
<ProcessStatistics
> *ProcStat
,
38 BitVector
*AffinityMask
) {
39 assert(Redirects
.empty() || Redirects
.size() == 3);
41 if (Execute(PI
, Program
, Args
, Env
, Redirects
, MemoryLimit
, ErrMsg
,
42 AffinityMask
, /*DetachProcess=*/false)) {
44 *ExecutionFailed
= false;
45 ProcessInfo Result
= Wait(
46 PI
, SecondsToWait
== 0 ? std::nullopt
: std::optional(SecondsToWait
),
48 return Result
.ReturnCode
;
52 *ExecutionFailed
= true;
57 ProcessInfo
sys::ExecuteNoWait(StringRef Program
, ArrayRef
<StringRef
> Args
,
58 std::optional
<ArrayRef
<StringRef
>> Env
,
59 ArrayRef
<std::optional
<StringRef
>> Redirects
,
60 unsigned MemoryLimit
, std::string
*ErrMsg
,
61 bool *ExecutionFailed
, BitVector
*AffinityMask
,
63 assert(Redirects
.empty() || Redirects
.size() == 3);
66 *ExecutionFailed
= false;
67 if (!Execute(PI
, Program
, Args
, Env
, Redirects
, MemoryLimit
, ErrMsg
,
68 AffinityMask
, DetachProcess
))
70 *ExecutionFailed
= true;
75 bool sys::commandLineFitsWithinSystemLimits(StringRef Program
,
76 ArrayRef
<const char *> Args
) {
77 SmallVector
<StringRef
, 8> StringRefArgs
;
78 StringRefArgs
.reserve(Args
.size());
79 for (const char *A
: Args
)
80 StringRefArgs
.emplace_back(A
);
81 return commandLineFitsWithinSystemLimits(Program
, StringRefArgs
);
84 void sys::printArg(raw_ostream
&OS
, StringRef Arg
, bool Quote
) {
85 const bool Escape
= Arg
.find_first_of(" \"\\$") != StringRef::npos
;
87 if (!Quote
&& !Escape
) {
92 // Quote and escape. This isn't really complete, but good enough.
94 for (const auto c
: Arg
) {
95 if (c
== '"' || c
== '\\' || c
== '$')
102 // Include the platform-specific parts of this class.
104 #include "Unix/Program.inc"
107 #include "Windows/Program.inc"