Files
swift-mirror/include/swift/Driver/Action.h
Jordan Rose 9c6ddc085f [driver] Add -lldb-repl and -integrated-repl modes.
Since LLDB is taking over as the REPL for Swift, we eventually want "swift"
and "swift -repl" to invoke "lldb --repl" rather than the frontend. However,
we only want to do this if the LLDB that's present is related to the Swift
that's present -- we don't want to invoke some random LLDB on the system
or in some other Xcode installation. Therefore, Swift searches for LLDB--
first next to the driver, then in the usr/bin/ outside of a toolchain--
before choosing to use it.

If the user just passes -repl and an LLDB is not found relative to the
driver, the existing "integrated" REPL will be launched instead.*

If the user passes -lldb-repl and an LLDB is not found relative to the
driver, one from the user's PATH will be chosen (like the linker).

The user can also pass -integrated-repl to get the existing behavior.
"swift -frontend -repl" always uses the integrated REPL.

* Since LLDB's not quite ready to be the REPL yet, "swift -repl" still
invokes the integrated REPL. "swift -repl -experimental-prefer-lldb" tests
the new behavior; this option will become the default (and the flag removed)
in <rdar://problem/16776719>.

<rdar://problem/16776705>

Swift SVN r17134
2014-05-01 02:15:53 +00:00

181 lines
4.5 KiB
C++

//===--- Action.h - Abstract compilation steps ------------------*- C++ -*-===//
//
// This source file is part of the Swift.org open source project
//
// Copyright (c) 2014 - 2015 Apple Inc. and the Swift project authors
// Licensed under Apache License v2.0 with Runtime Library Exception
//
// See http://swift.org/LICENSE.txt for license information
// See http://swift.org/CONTRIBUTORS.txt for the list of Swift project authors
//
//===----------------------------------------------------------------------===//
#ifndef SWIFT_DRIVER_ACTION_H
#define SWIFT_DRIVER_ACTION_H
#include "swift/Basic/LLVM.h"
#include "swift/Driver/Types.h"
#include "swift/Driver/Util.h"
#include "llvm/ADT/ArrayRef.h"
namespace llvm {
namespace opt {
class Arg;
}
}
namespace swift {
namespace driver {
class Action;
class Action {
public:
typedef ActionList::size_type size_type;
typedef ActionList::iterator iterator;
typedef ActionList::const_iterator const_iterator;
enum ActionClass {
Input = 0,
CompileJob,
MergeModuleJob,
REPLJob,
LinkJob,
JobFirst=CompileJob,
JobLast=LinkJob
};
static const char *getClassName(ActionClass AC);
private:
ActionClass Kind;
types::ID Type;
ActionList Inputs;
unsigned OwnsInputs : 1;
protected:
Action(ActionClass Kind, types::ID Type)
: Kind(Kind), Type(Type), OwnsInputs(true) {}
Action(ActionClass Kind, ArrayRef<Action *> Inputs, types::ID Type)
: Kind(Kind), Type(Type), Inputs(Inputs.begin(), Inputs.end()),
OwnsInputs(true) {}
public:
virtual ~Action();
const char *getClassName() const { return Action::getClassName(getKind()); }
bool getOwnsInputs() const { return OwnsInputs; }
void setOwnsInputs(bool Value) { OwnsInputs = Value; }
ActionClass getKind() const { return Kind; }
types::ID getType() const { return Type; }
ArrayRef<Action *> getInputs() const { return Inputs; }
void addInput(Action *Input) { Inputs.push_back(Input); }
size_type size() const { return Inputs.size(); }
iterator begin() { return Inputs.begin(); }
iterator end() { return Inputs.end(); }
const_iterator begin() const { return Inputs.begin(); }
const_iterator end() const { return Inputs.end(); }
};
class InputAction : public Action {
virtual void anchor();
const llvm::opt::Arg &Input;
public:
InputAction(const llvm::opt::Arg &Input, types::ID Type)
: Action(Action::Input, Type), Input(Input) {}
const llvm::opt::Arg &getInputArg() const { return Input; }
static bool classof(const Action *A) {
return A->getKind() == Action::Input;
}
};
class JobAction : public Action {
virtual void anchor();
protected:
JobAction(ActionClass Kind, ArrayRef<Action *> Inputs, types::ID Type)
: Action(Kind, Inputs, Type) {}
public:
static bool classof(const Action *A) {
return (A->getKind() >= ActionClass::JobFirst &&
A->getKind() <= ActionClass::JobLast);
}
};
class CompileJobAction : public JobAction {
virtual void anchor();
public:
CompileJobAction(types::ID OutputType)
: JobAction(Action::CompileJob, llvm::None, OutputType) {}
CompileJobAction(Action *Input, types::ID OutputType)
: JobAction(Action::CompileJob, Input, OutputType) {}
static bool classof(const Action *A) {
return A->getKind() == Action::CompileJob;
}
};
class REPLJobAction : public JobAction {
public:
enum class Mode {
Integrated,
PreferLLDB,
RequireLLDB
};
private:
virtual void anchor();
Mode RequestedMode;
public:
REPLJobAction(Mode mode)
: JobAction(Action::REPLJob, llvm::None, types::TY_Nothing),
RequestedMode(mode) {}
Mode getRequestedMode() const { return RequestedMode; }
static bool classof(const Action *A) {
return A->getKind() == Action::REPLJob;
}
};
class MergeModuleJobAction : public JobAction {
virtual void anchor();
public:
MergeModuleJobAction(ArrayRef<Action *> Inputs)
: JobAction(Action::MergeModuleJob, Inputs, types::TY_SwiftModuleFile) {}
static bool classof(const Action *A) {
return A->getKind() == Action::MergeModuleJob;
}
};
class LinkJobAction : public JobAction {
virtual void anchor();
LinkKind Kind;
public:
LinkJobAction(ArrayRef<Action *> Inputs, LinkKind K)
: JobAction(Action::LinkJob, Inputs, types::TY_Image), Kind(K) {
assert(Kind != LinkKind::None);
}
LinkKind getKind() const { return Kind; }
static bool classof(const Action *A) {
return A->getKind() == Action::LinkJob;
}
};
} // end namespace driver
} // end namespace swift
#endif