2001-08-23 17:05:04 +00:00
|
|
|
//===-- Support.cpp - Support routines for interpreter --------------------===//
|
|
|
|
//
|
|
|
|
// This file contains support routines for the interpreter core.
|
|
|
|
//
|
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
|
|
|
|
#include "Interpreter.h"
|
|
|
|
#include "llvm/SymbolTable.h"
|
|
|
|
#include "llvm/Assembly/Writer.h"
|
2002-01-20 22:54:45 +00:00
|
|
|
#include <iostream>
|
|
|
|
using std::cout;
|
2001-08-23 17:05:04 +00:00
|
|
|
|
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
//
|
|
|
|
// LookupMatchingNames helper - Search a symbol table for values matching Name.
|
|
|
|
//
|
2002-01-20 22:54:45 +00:00
|
|
|
static inline void LookupMatchingNames(const std::string &Name,SymTabValue &STV,
|
|
|
|
std::vector<Value*> &Results) {
|
2001-08-23 17:05:04 +00:00
|
|
|
SymbolTable *SymTab = STV.getSymbolTable();
|
|
|
|
if (SymTab == 0) return; // No symbolic values :(
|
|
|
|
|
|
|
|
// Loop over all of the type planes in the symbol table...
|
|
|
|
for (SymbolTable::iterator I = SymTab->begin(), E = SymTab->end();
|
|
|
|
I != E; ++I) {
|
|
|
|
SymbolTable::VarMap &Plane = I->second;
|
|
|
|
|
|
|
|
// Search the symbol table plane for this name...
|
|
|
|
SymbolTable::VarMap::iterator Val = Plane.find(Name);
|
|
|
|
if (Val != Plane.end())
|
|
|
|
Results.push_back(Val->second); // Found a name match!
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// LookupMatchingNames - Search the current method namespace, then the global
|
|
|
|
// namespace looking for values that match the specified name. Return ALL
|
|
|
|
// matches to that name. This is obviously slow, and should only be used for
|
|
|
|
// user interaction.
|
|
|
|
//
|
2002-01-20 22:54:45 +00:00
|
|
|
std::vector<Value*> Interpreter::LookupMatchingNames(const std::string &Name) {
|
|
|
|
std::vector<Value*> Results;
|
2001-08-23 17:05:04 +00:00
|
|
|
Method *CurMeth = getCurrentMethod();
|
|
|
|
|
|
|
|
if (CurMeth) ::LookupMatchingNames(Name, *CurMeth, Results);
|
|
|
|
if (CurMod ) ::LookupMatchingNames(Name, *CurMod , Results);
|
|
|
|
return Results;
|
|
|
|
}
|
|
|
|
|
|
|
|
// ChooseOneOption - Prompt the user to choose among the specified options to
|
|
|
|
// pick one value. If no options are provided, emit an error. If a single
|
|
|
|
// option is provided, just return that option.
|
|
|
|
//
|
2002-01-20 22:54:45 +00:00
|
|
|
Value *Interpreter::ChooseOneOption(const std::string &Name,
|
|
|
|
const std::vector<Value*> &Opts) {
|
2001-08-23 17:05:04 +00:00
|
|
|
switch (Opts.size()) {
|
|
|
|
case 1: return Opts[0];
|
|
|
|
case 0:
|
|
|
|
cout << "Error: no entities named '" << Name << "' found!\n";
|
|
|
|
return 0;
|
|
|
|
default: break; // Must prompt user...
|
|
|
|
}
|
|
|
|
|
|
|
|
cout << "Multiple entities named '" << Name << "' found! Please choose:\n";
|
|
|
|
cout << " 0. Cancel operation\n";
|
|
|
|
for (unsigned i = 0; i < Opts.size(); ++i) {
|
|
|
|
cout << " " << (i+1) << ".";
|
2002-01-20 22:54:45 +00:00
|
|
|
WriteAsOperand(cout, Opts[i]) << "\n";
|
2001-08-23 17:05:04 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
unsigned Option;
|
|
|
|
do {
|
2002-01-20 22:54:45 +00:00
|
|
|
cout << "lli> " << std::flush;
|
|
|
|
std::cin >> Option;
|
2001-08-23 17:05:04 +00:00
|
|
|
if (Option > Opts.size())
|
|
|
|
cout << "Invalid selection: Please choose from 0 to " << Opts.size()
|
2002-01-20 22:54:45 +00:00
|
|
|
<< "\n";
|
2001-08-23 17:05:04 +00:00
|
|
|
} while (Option > Opts.size());
|
|
|
|
|
|
|
|
if (Option == 0) return 0;
|
|
|
|
return Opts[Option-1];
|
|
|
|
}
|