2018-06-23 01:39:19 +08:00
|
|
|
//===- Parser.cpp - MLIR Parser Implementation ----------------------------===//
|
|
|
|
//
|
|
|
|
// Copyright 2019 The MLIR Authors.
|
|
|
|
//
|
|
|
|
// Licensed under the Apache License, Version 2.0 (the "License");
|
|
|
|
// you may not use this file except in compliance with the License.
|
|
|
|
// You may obtain a copy of the License at
|
|
|
|
//
|
|
|
|
// http://www.apache.org/licenses/LICENSE-2.0
|
|
|
|
//
|
|
|
|
// Unless required by applicable law or agreed to in writing, software
|
|
|
|
// distributed under the License is distributed on an "AS IS" BASIS,
|
|
|
|
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
|
|
// See the License for the specific language governing permissions and
|
|
|
|
// limitations under the License.
|
|
|
|
// =============================================================================
|
|
|
|
//
|
|
|
|
// This file implements the parser for the MLIR textual form.
|
|
|
|
//
|
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
|
|
|
|
#include "mlir/Parser.h"
|
|
|
|
#include "Lexer.h"
|
2018-06-30 09:09:29 +08:00
|
|
|
#include "mlir/IR/AffineExpr.h"
|
2018-06-28 02:03:08 +08:00
|
|
|
#include "mlir/IR/AffineMap.h"
|
2018-07-05 11:45:39 +08:00
|
|
|
#include "mlir/IR/Attributes.h"
|
2018-07-09 11:51:38 +08:00
|
|
|
#include "mlir/IR/Builders.h"
|
2018-06-29 08:02:32 +08:00
|
|
|
#include "mlir/IR/MLFunction.h"
|
2018-08-21 23:42:19 +08:00
|
|
|
#include "mlir/IR/MLIRContext.h"
|
2018-07-07 01:46:19 +08:00
|
|
|
#include "mlir/IR/Module.h"
|
2018-07-26 02:15:20 +08:00
|
|
|
#include "mlir/IR/OpImplementation.h"
|
2018-07-07 01:46:19 +08:00
|
|
|
#include "mlir/IR/OperationSet.h"
|
2018-08-21 23:42:19 +08:00
|
|
|
#include "mlir/IR/StmtVisitor.h"
|
2018-06-23 13:03:48 +08:00
|
|
|
#include "mlir/IR/Types.h"
|
2018-08-03 16:54:46 +08:00
|
|
|
#include "llvm/ADT/DenseMap.h"
|
2018-08-22 08:55:22 +08:00
|
|
|
#include "llvm/Support/PrettyStackTrace.h"
|
2018-08-03 16:54:46 +08:00
|
|
|
#include "llvm/Support/SourceMgr.h"
|
2018-06-23 01:39:19 +08:00
|
|
|
using namespace mlir;
|
2018-06-24 07:03:42 +08:00
|
|
|
using llvm::SMLoc;
|
2018-07-24 07:56:32 +08:00
|
|
|
using llvm::SourceMgr;
|
2018-06-23 01:39:19 +08:00
|
|
|
|
2018-06-23 13:03:48 +08:00
|
|
|
/// Simple enum to make code read better in cases that would otherwise return a
|
|
|
|
/// bool value. Failure is "true" in a boolean context.
|
2018-07-24 07:56:32 +08:00
|
|
|
enum ParseResult { ParseSuccess, ParseFailure };
|
2018-06-23 01:39:19 +08:00
|
|
|
|
2018-07-10 10:05:38 +08:00
|
|
|
namespace {
|
|
|
|
class Parser;
|
|
|
|
|
|
|
|
/// This class refers to all of the state maintained globally by the parser,
|
|
|
|
/// such as the current lexer position etc. The Parser base class provides
|
|
|
|
/// methods to access this.
|
|
|
|
class ParserState {
|
2018-06-29 11:45:33 +08:00
|
|
|
public:
|
2018-07-11 01:08:27 +08:00
|
|
|
ParserState(llvm::SourceMgr &sourceMgr, Module *module,
|
2018-07-10 10:05:38 +08:00
|
|
|
SMDiagnosticHandlerTy errorReporter)
|
2018-07-11 01:08:27 +08:00
|
|
|
: context(module->getContext()), module(module),
|
|
|
|
lex(sourceMgr, errorReporter), curToken(lex.lexToken()),
|
2018-07-26 02:15:20 +08:00
|
|
|
errorReporter(errorReporter), operationSet(OperationSet::get(context)) {
|
|
|
|
}
|
2018-07-11 01:08:27 +08:00
|
|
|
|
|
|
|
// A map from affine map identifier to AffineMap.
|
|
|
|
llvm::StringMap<AffineMap *> affineMapDefinitions;
|
2018-08-20 12:17:22 +08:00
|
|
|
|
2018-08-08 05:24:38 +08:00
|
|
|
// A map from integer set identifier to IntegerSet.
|
|
|
|
llvm::StringMap<IntegerSet *> integerSetDefinitions;
|
2018-06-23 01:39:19 +08:00
|
|
|
|
2018-08-20 12:17:22 +08:00
|
|
|
// This keeps track of all forward references to functions along with the
|
|
|
|
// temporary function used to represent them and the location of the first
|
|
|
|
// reference.
|
|
|
|
llvm::DenseMap<Identifier, std::pair<Function *, SMLoc>> functionForwardRefs;
|
|
|
|
|
2018-06-23 01:39:19 +08:00
|
|
|
private:
|
2018-07-10 10:05:38 +08:00
|
|
|
ParserState(const ParserState &) = delete;
|
|
|
|
void operator=(const ParserState &) = delete;
|
|
|
|
|
|
|
|
friend class Parser;
|
|
|
|
|
|
|
|
// The context we're parsing into.
|
2018-07-11 01:08:27 +08:00
|
|
|
MLIRContext *const context;
|
|
|
|
|
|
|
|
// This is the module we are parsing into.
|
|
|
|
Module *const module;
|
2018-06-23 13:03:48 +08:00
|
|
|
|
|
|
|
// The lexer for the source file we're parsing.
|
2018-06-23 01:39:19 +08:00
|
|
|
Lexer lex;
|
|
|
|
|
|
|
|
// This is the next token that hasn't been consumed yet.
|
|
|
|
Token curToken;
|
|
|
|
|
2018-06-25 10:17:35 +08:00
|
|
|
// The diagnostic error reporter.
|
2018-07-11 01:08:27 +08:00
|
|
|
SMDiagnosticHandlerTy const errorReporter;
|
2018-07-26 02:15:20 +08:00
|
|
|
|
|
|
|
// The active OperationSet we're parsing with.
|
|
|
|
OperationSet &operationSet;
|
2018-07-10 10:05:38 +08:00
|
|
|
};
|
|
|
|
} // end anonymous namespace
|
|
|
|
|
|
|
|
namespace {
|
|
|
|
|
2018-08-08 03:02:37 +08:00
|
|
|
typedef std::function<Operation *(const OperationState &)>
|
2018-07-17 02:47:09 +08:00
|
|
|
CreateOperationFunction;
|
|
|
|
|
2018-07-10 10:05:38 +08:00
|
|
|
/// This class implement support for parsing global entities like types and
|
|
|
|
/// shared entities like SSA names. It is intended to be subclassed by
|
|
|
|
/// specialized subparsers that include state, e.g. when a local symbol table.
|
|
|
|
class Parser {
|
|
|
|
public:
|
2018-07-11 01:08:27 +08:00
|
|
|
Builder builder;
|
|
|
|
|
|
|
|
Parser(ParserState &state) : builder(state.context), state(state) {}
|
2018-06-28 02:03:08 +08:00
|
|
|
|
2018-07-11 01:08:27 +08:00
|
|
|
// Helper methods to get stuff from the parser-global state.
|
|
|
|
ParserState &getState() const { return state; }
|
2018-07-10 10:05:38 +08:00
|
|
|
MLIRContext *getContext() const { return state.context; }
|
2018-07-11 01:08:27 +08:00
|
|
|
Module *getModule() { return state.module; }
|
2018-07-26 02:15:20 +08:00
|
|
|
OperationSet &getOperationSet() const { return state.operationSet; }
|
2018-08-06 12:12:29 +08:00
|
|
|
llvm::SourceMgr &getSourceMgr() { return state.lex.getSourceMgr(); }
|
2018-07-10 10:05:38 +08:00
|
|
|
|
|
|
|
/// Return the current token the parser is inspecting.
|
|
|
|
const Token &getToken() const { return state.curToken; }
|
|
|
|
StringRef getTokenSpelling() const { return state.curToken.getSpelling(); }
|
2018-06-23 01:39:19 +08:00
|
|
|
|
2018-08-24 05:32:25 +08:00
|
|
|
/// Encode the specified source location information into an attribute for
|
|
|
|
/// attachment to the IR.
|
|
|
|
Attribute *getEncodedSourceLocation(llvm::SMLoc loc);
|
|
|
|
|
2018-06-23 01:39:19 +08:00
|
|
|
/// Emit an error and return failure.
|
2018-06-24 07:03:42 +08:00
|
|
|
ParseResult emitError(const Twine &message) {
|
2018-07-10 10:05:38 +08:00
|
|
|
return emitError(state.curToken.getLoc(), message);
|
2018-06-24 07:03:42 +08:00
|
|
|
}
|
|
|
|
ParseResult emitError(SMLoc loc, const Twine &message);
|
2018-06-23 01:39:19 +08:00
|
|
|
|
|
|
|
/// Advance the current lexer onto the next token.
|
|
|
|
void consumeToken() {
|
2018-07-10 10:05:38 +08:00
|
|
|
assert(state.curToken.isNot(Token::eof, Token::error) &&
|
2018-06-23 01:39:19 +08:00
|
|
|
"shouldn't advance past EOF or errors");
|
2018-07-10 10:05:38 +08:00
|
|
|
state.curToken = state.lex.lexToken();
|
2018-06-23 01:39:19 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
/// Advance the current lexer onto the next token, asserting what the expected
|
|
|
|
/// current token is. This is preferred to the above method because it leads
|
|
|
|
/// to more self-documenting code with better checking.
|
2018-06-30 02:15:56 +08:00
|
|
|
void consumeToken(Token::Kind kind) {
|
2018-07-10 10:05:38 +08:00
|
|
|
assert(state.curToken.is(kind) && "consumed an unexpected token");
|
2018-06-23 01:39:19 +08:00
|
|
|
consumeToken();
|
|
|
|
}
|
|
|
|
|
2018-06-23 06:52:02 +08:00
|
|
|
/// If the current token has the specified kind, consume it and return true.
|
|
|
|
/// If not, return false.
|
2018-06-30 02:15:56 +08:00
|
|
|
bool consumeIf(Token::Kind kind) {
|
2018-07-10 10:05:38 +08:00
|
|
|
if (state.curToken.isNot(kind))
|
2018-06-23 06:52:02 +08:00
|
|
|
return false;
|
|
|
|
consumeToken(kind);
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2018-07-24 08:30:01 +08:00
|
|
|
/// Consume the specified token if present and return success. On failure,
|
|
|
|
/// output a diagnostic and return failure.
|
|
|
|
ParseResult parseToken(Token::Kind expectedToken, const Twine &message);
|
|
|
|
|
2018-07-22 05:32:09 +08:00
|
|
|
/// Parse a comma-separated list of elements up until the specified end token.
|
|
|
|
ParseResult
|
|
|
|
parseCommaSeparatedListUntil(Token::Kind rightToken,
|
|
|
|
const std::function<ParseResult()> &parseElement,
|
|
|
|
bool allowEmptyList = true);
|
|
|
|
|
|
|
|
/// Parse a comma separated list of elements that must have at least one entry
|
|
|
|
/// in it.
|
|
|
|
ParseResult
|
|
|
|
parseCommaSeparatedList(const std::function<ParseResult()> &parseElement);
|
2018-06-23 06:52:02 +08:00
|
|
|
|
2018-06-23 13:03:48 +08:00
|
|
|
// We have two forms of parsing methods - those that return a non-null
|
|
|
|
// pointer on success, and those that return a ParseResult to indicate whether
|
|
|
|
// they returned a failure. The second class fills in by-reference arguments
|
|
|
|
// as the results of their action.
|
|
|
|
|
2018-06-23 01:39:19 +08:00
|
|
|
// Type parsing.
|
2018-06-23 13:03:48 +08:00
|
|
|
VectorType *parseVectorType();
|
2018-06-23 06:52:02 +08:00
|
|
|
ParseResult parseDimensionListRanked(SmallVectorImpl<int> &dimensions);
|
2018-06-23 13:03:48 +08:00
|
|
|
Type *parseTensorType();
|
|
|
|
Type *parseMemRefType();
|
|
|
|
Type *parseFunctionType();
|
|
|
|
Type *parseType();
|
2018-07-23 23:42:19 +08:00
|
|
|
ParseResult parseTypeListNoParens(SmallVectorImpl<Type *> &elements);
|
2018-07-24 07:56:32 +08:00
|
|
|
ParseResult parseTypeList(SmallVectorImpl<Type *> &elements);
|
2018-06-23 01:39:19 +08:00
|
|
|
|
2018-07-05 11:45:39 +08:00
|
|
|
// Attribute parsing.
|
2018-08-22 08:55:22 +08:00
|
|
|
Function *resolveFunctionReference(StringRef nameStr, SMLoc nameLoc,
|
|
|
|
FunctionType *type);
|
2018-07-05 11:45:39 +08:00
|
|
|
Attribute *parseAttribute();
|
|
|
|
ParseResult parseAttributeDict(SmallVectorImpl<NamedAttribute> &attributes);
|
|
|
|
|
2018-07-04 11:16:08 +08:00
|
|
|
// Polyhedral structures.
|
2018-07-11 01:08:27 +08:00
|
|
|
AffineMap *parseAffineMapInline();
|
2018-07-17 00:45:22 +08:00
|
|
|
AffineMap *parseAffineMapReference();
|
2018-08-08 05:24:38 +08:00
|
|
|
IntegerSet *parseIntegerSetInline();
|
|
|
|
IntegerSet *parseIntegerSetReference();
|
2018-06-28 02:03:08 +08:00
|
|
|
|
2018-07-10 10:05:38 +08:00
|
|
|
private:
|
|
|
|
// The Parser is subclassed and reinstantiated. Do not add additional
|
|
|
|
// non-trivial state here, add it to the ParserState class.
|
|
|
|
ParserState &state;
|
2018-06-23 01:39:19 +08:00
|
|
|
};
|
|
|
|
} // end anonymous namespace
|
|
|
|
|
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
// Helper methods.
|
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
|
2018-08-24 05:32:25 +08:00
|
|
|
/// Encode the specified source location information into an attribute for
|
|
|
|
/// attachment to the IR.
|
|
|
|
Attribute *Parser::getEncodedSourceLocation(llvm::SMLoc loc) {
|
|
|
|
// TODO(clattner): Switch to an more structured form that includes
|
|
|
|
// file/line/column instead of just byte offset in the file. This will
|
|
|
|
// eliminate this block of low level code poking at the SourceMgr directly.
|
|
|
|
auto &sourceMgr = getSourceMgr();
|
|
|
|
auto fileID = sourceMgr.FindBufferContainingLoc(loc);
|
|
|
|
|
|
|
|
auto *srcBuffer = sourceMgr.getMemoryBuffer(fileID);
|
|
|
|
unsigned locationEncoding = loc.getPointer() - srcBuffer->getBufferStart();
|
|
|
|
return builder.getIntegerAttr(locationEncoding);
|
|
|
|
}
|
|
|
|
|
2018-06-24 07:03:42 +08:00
|
|
|
ParseResult Parser::emitError(SMLoc loc, const Twine &message) {
|
2018-06-23 06:52:02 +08:00
|
|
|
// If we hit a parse error in response to a lexer error, then the lexer
|
2018-06-25 10:17:35 +08:00
|
|
|
// already reported the error.
|
2018-07-10 10:05:38 +08:00
|
|
|
if (getToken().is(Token::error))
|
2018-06-23 06:52:02 +08:00
|
|
|
return ParseFailure;
|
|
|
|
|
2018-07-10 10:05:38 +08:00
|
|
|
auto &sourceMgr = state.lex.getSourceMgr();
|
|
|
|
state.errorReporter(sourceMgr.GetMessage(loc, SourceMgr::DK_Error, message));
|
2018-06-23 01:39:19 +08:00
|
|
|
return ParseFailure;
|
|
|
|
}
|
|
|
|
|
2018-07-24 08:30:01 +08:00
|
|
|
/// Consume the specified token if present and return success. On failure,
|
|
|
|
/// output a diagnostic and return failure.
|
|
|
|
ParseResult Parser::parseToken(Token::Kind expectedToken,
|
|
|
|
const Twine &message) {
|
|
|
|
if (consumeIf(expectedToken))
|
|
|
|
return ParseSuccess;
|
|
|
|
return emitError(message);
|
|
|
|
}
|
|
|
|
|
2018-07-22 05:32:09 +08:00
|
|
|
/// Parse a comma separated list of elements that must have at least one entry
|
|
|
|
/// in it.
|
|
|
|
ParseResult Parser::parseCommaSeparatedList(
|
|
|
|
const std::function<ParseResult()> &parseElement) {
|
|
|
|
// Non-empty case starts with an element.
|
|
|
|
if (parseElement())
|
|
|
|
return ParseFailure;
|
|
|
|
|
|
|
|
// Otherwise we have a list of comma separated elements.
|
|
|
|
while (consumeIf(Token::comma)) {
|
|
|
|
if (parseElement())
|
|
|
|
return ParseFailure;
|
|
|
|
}
|
|
|
|
return ParseSuccess;
|
|
|
|
}
|
|
|
|
|
2018-06-23 06:52:02 +08:00
|
|
|
/// Parse a comma-separated list of elements, terminated with an arbitrary
|
|
|
|
/// token. This allows empty lists if allowEmptyList is true.
|
|
|
|
///
|
|
|
|
/// abstract-list ::= rightToken // if allowEmptyList == true
|
|
|
|
/// abstract-list ::= element (',' element)* rightToken
|
|
|
|
///
|
2018-07-22 05:32:09 +08:00
|
|
|
ParseResult Parser::parseCommaSeparatedListUntil(
|
|
|
|
Token::Kind rightToken, const std::function<ParseResult()> &parseElement,
|
|
|
|
bool allowEmptyList) {
|
2018-06-23 06:52:02 +08:00
|
|
|
// Handle the empty case.
|
2018-07-10 10:05:38 +08:00
|
|
|
if (getToken().is(rightToken)) {
|
2018-06-23 06:52:02 +08:00
|
|
|
if (!allowEmptyList)
|
|
|
|
return emitError("expected list element");
|
|
|
|
consumeToken(rightToken);
|
|
|
|
return ParseSuccess;
|
|
|
|
}
|
|
|
|
|
2018-07-24 08:30:01 +08:00
|
|
|
if (parseCommaSeparatedList(parseElement) ||
|
|
|
|
parseToken(rightToken, "expected ',' or '" +
|
|
|
|
Token::getTokenSpelling(rightToken) + "'"))
|
2018-06-23 06:52:02 +08:00
|
|
|
return ParseFailure;
|
|
|
|
|
|
|
|
return ParseSuccess;
|
|
|
|
}
|
2018-06-23 01:39:19 +08:00
|
|
|
|
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
// Type Parsing
|
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
|
Eliminate "primitive" types from being a thing, splitting them into FloatType
and OtherType. Other type is now the thing that holds AffineInt, Control,
eventually Resource, Variant, String, etc. FloatType holds the floating point
types, and allows convenient query of isa<FloatType>().
This fixes issues where we allowed control to be the element type of tensor,
memref, vector. At the same time, ban AffineInt from being an element of a
vector/memref/tensor as well since we don't need it.
I updated the spec to match this as well.
PiperOrigin-RevId: 206361942
2018-07-28 04:09:58 +08:00
|
|
|
/// Parse an arbitrary type.
|
|
|
|
///
|
|
|
|
/// type ::= integer-type
|
|
|
|
/// | float-type
|
|
|
|
/// | other-type
|
|
|
|
/// | vector-type
|
|
|
|
/// | tensor-type
|
|
|
|
/// | memref-type
|
|
|
|
/// | function-type
|
2018-06-23 06:52:02 +08:00
|
|
|
///
|
Eliminate "primitive" types from being a thing, splitting them into FloatType
and OtherType. Other type is now the thing that holds AffineInt, Control,
eventually Resource, Variant, String, etc. FloatType holds the floating point
types, and allows convenient query of isa<FloatType>().
This fixes issues where we allowed control to be the element type of tensor,
memref, vector. At the same time, ban AffineInt from being an element of a
vector/memref/tensor as well since we don't need it.
I updated the spec to match this as well.
PiperOrigin-RevId: 206361942
2018-07-28 04:09:58 +08:00
|
|
|
/// float-type ::= `f16` | `bf16` | `f32` | `f64`
|
|
|
|
/// other-type ::= `affineint` | `tf_control`
|
2018-06-23 06:52:02 +08:00
|
|
|
///
|
Eliminate "primitive" types from being a thing, splitting them into FloatType
and OtherType. Other type is now the thing that holds AffineInt, Control,
eventually Resource, Variant, String, etc. FloatType holds the floating point
types, and allows convenient query of isa<FloatType>().
This fixes issues where we allowed control to be the element type of tensor,
memref, vector. At the same time, ban AffineInt from being an element of a
vector/memref/tensor as well since we don't need it.
I updated the spec to match this as well.
PiperOrigin-RevId: 206361942
2018-07-28 04:09:58 +08:00
|
|
|
Type *Parser::parseType() {
|
2018-07-10 10:05:38 +08:00
|
|
|
switch (getToken().getKind()) {
|
2018-06-23 13:03:48 +08:00
|
|
|
default:
|
|
|
|
return (emitError("expected type"), nullptr);
|
Eliminate "primitive" types from being a thing, splitting them into FloatType
and OtherType. Other type is now the thing that holds AffineInt, Control,
eventually Resource, Variant, String, etc. FloatType holds the floating point
types, and allows convenient query of isa<FloatType>().
This fixes issues where we allowed control to be the element type of tensor,
memref, vector. At the same time, ban AffineInt from being an element of a
vector/memref/tensor as well since we don't need it.
I updated the spec to match this as well.
PiperOrigin-RevId: 206361942
2018-07-28 04:09:58 +08:00
|
|
|
case Token::kw_memref:
|
|
|
|
return parseMemRefType();
|
|
|
|
case Token::kw_tensor:
|
|
|
|
return parseTensorType();
|
|
|
|
case Token::kw_vector:
|
|
|
|
return parseVectorType();
|
|
|
|
case Token::l_paren:
|
|
|
|
return parseFunctionType();
|
|
|
|
// integer-type
|
|
|
|
case Token::inttype: {
|
|
|
|
auto width = getToken().getIntTypeBitwidth();
|
|
|
|
if (!width.hasValue())
|
|
|
|
return (emitError("invalid integer width"), nullptr);
|
|
|
|
consumeToken(Token::inttype);
|
|
|
|
return builder.getIntegerType(width.getValue());
|
|
|
|
}
|
|
|
|
|
|
|
|
// float-type
|
2018-06-23 06:52:02 +08:00
|
|
|
case Token::kw_bf16:
|
|
|
|
consumeToken(Token::kw_bf16);
|
2018-07-09 11:51:38 +08:00
|
|
|
return builder.getBF16Type();
|
2018-06-23 06:52:02 +08:00
|
|
|
case Token::kw_f16:
|
|
|
|
consumeToken(Token::kw_f16);
|
2018-07-09 11:51:38 +08:00
|
|
|
return builder.getF16Type();
|
2018-06-23 06:52:02 +08:00
|
|
|
case Token::kw_f32:
|
|
|
|
consumeToken(Token::kw_f32);
|
2018-07-09 11:51:38 +08:00
|
|
|
return builder.getF32Type();
|
2018-06-23 06:52:02 +08:00
|
|
|
case Token::kw_f64:
|
|
|
|
consumeToken(Token::kw_f64);
|
2018-07-09 11:51:38 +08:00
|
|
|
return builder.getF64Type();
|
Eliminate "primitive" types from being a thing, splitting them into FloatType
and OtherType. Other type is now the thing that holds AffineInt, Control,
eventually Resource, Variant, String, etc. FloatType holds the floating point
types, and allows convenient query of isa<FloatType>().
This fixes issues where we allowed control to be the element type of tensor,
memref, vector. At the same time, ban AffineInt from being an element of a
vector/memref/tensor as well since we don't need it.
I updated the spec to match this as well.
PiperOrigin-RevId: 206361942
2018-07-28 04:09:58 +08:00
|
|
|
|
|
|
|
// other-type
|
2018-06-30 13:08:05 +08:00
|
|
|
case Token::kw_affineint:
|
|
|
|
consumeToken(Token::kw_affineint);
|
2018-07-09 11:51:38 +08:00
|
|
|
return builder.getAffineIntType();
|
2018-07-28 02:07:12 +08:00
|
|
|
case Token::kw_tf_control:
|
|
|
|
consumeToken(Token::kw_tf_control);
|
|
|
|
return builder.getTFControlType();
|
2018-08-02 03:55:27 +08:00
|
|
|
case Token::kw_tf_string:
|
|
|
|
consumeToken(Token::kw_tf_string);
|
|
|
|
return builder.getTFStringType();
|
2018-06-23 06:52:02 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Parse a vector type.
|
|
|
|
///
|
|
|
|
/// vector-type ::= `vector` `<` const-dimension-list primitive-type `>`
|
|
|
|
/// const-dimension-list ::= (integer-literal `x`)+
|
|
|
|
///
|
2018-06-23 13:03:48 +08:00
|
|
|
VectorType *Parser::parseVectorType() {
|
2018-06-23 06:52:02 +08:00
|
|
|
consumeToken(Token::kw_vector);
|
|
|
|
|
2018-07-24 08:30:01 +08:00
|
|
|
if (parseToken(Token::less, "expected '<' in vector type"))
|
|
|
|
return nullptr;
|
2018-06-23 06:52:02 +08:00
|
|
|
|
2018-07-10 10:05:38 +08:00
|
|
|
if (getToken().isNot(Token::integer))
|
2018-06-23 13:03:48 +08:00
|
|
|
return (emitError("expected dimension size in vector type"), nullptr);
|
2018-06-23 06:52:02 +08:00
|
|
|
|
|
|
|
SmallVector<unsigned, 4> dimensions;
|
2018-07-10 10:05:38 +08:00
|
|
|
while (getToken().is(Token::integer)) {
|
2018-06-23 06:52:02 +08:00
|
|
|
// Make sure this integer value is in bound and valid.
|
2018-07-10 10:05:38 +08:00
|
|
|
auto dimension = getToken().getUnsignedIntegerValue();
|
2018-06-23 06:52:02 +08:00
|
|
|
if (!dimension.hasValue())
|
2018-06-23 13:03:48 +08:00
|
|
|
return (emitError("invalid dimension in vector type"), nullptr);
|
2018-06-23 06:52:02 +08:00
|
|
|
dimensions.push_back(dimension.getValue());
|
|
|
|
|
|
|
|
consumeToken(Token::integer);
|
|
|
|
|
|
|
|
// Make sure we have an 'x' or something like 'xbf32'.
|
2018-07-10 10:05:38 +08:00
|
|
|
if (getToken().isNot(Token::bare_identifier) ||
|
|
|
|
getTokenSpelling()[0] != 'x')
|
2018-06-23 13:03:48 +08:00
|
|
|
return (emitError("expected 'x' in vector dimension list"), nullptr);
|
2018-06-23 06:52:02 +08:00
|
|
|
|
|
|
|
// If we had a prefix of 'x', lex the next token immediately after the 'x'.
|
2018-07-10 10:05:38 +08:00
|
|
|
if (getTokenSpelling().size() != 1)
|
|
|
|
state.lex.resetPointer(getTokenSpelling().data() + 1);
|
2018-06-23 06:52:02 +08:00
|
|
|
|
|
|
|
// Consume the 'x'.
|
|
|
|
consumeToken(Token::bare_identifier);
|
|
|
|
}
|
|
|
|
|
|
|
|
// Parse the element type.
|
Eliminate "primitive" types from being a thing, splitting them into FloatType
and OtherType. Other type is now the thing that holds AffineInt, Control,
eventually Resource, Variant, String, etc. FloatType holds the floating point
types, and allows convenient query of isa<FloatType>().
This fixes issues where we allowed control to be the element type of tensor,
memref, vector. At the same time, ban AffineInt from being an element of a
vector/memref/tensor as well since we don't need it.
I updated the spec to match this as well.
PiperOrigin-RevId: 206361942
2018-07-28 04:09:58 +08:00
|
|
|
auto typeLoc = getToken().getLoc();
|
|
|
|
auto *elementType = parseType();
|
|
|
|
if (!elementType || parseToken(Token::greater, "expected '>' in vector type"))
|
2018-06-23 13:03:48 +08:00
|
|
|
return nullptr;
|
2018-06-23 06:52:02 +08:00
|
|
|
|
Eliminate "primitive" types from being a thing, splitting them into FloatType
and OtherType. Other type is now the thing that holds AffineInt, Control,
eventually Resource, Variant, String, etc. FloatType holds the floating point
types, and allows convenient query of isa<FloatType>().
This fixes issues where we allowed control to be the element type of tensor,
memref, vector. At the same time, ban AffineInt from being an element of a
vector/memref/tensor as well since we don't need it.
I updated the spec to match this as well.
PiperOrigin-RevId: 206361942
2018-07-28 04:09:58 +08:00
|
|
|
if (!isa<FloatType>(elementType) && !isa<IntegerType>(elementType))
|
|
|
|
return (emitError(typeLoc, "invalid vector element type"), nullptr);
|
2018-06-23 06:52:02 +08:00
|
|
|
|
2018-06-23 13:03:48 +08:00
|
|
|
return VectorType::get(dimensions, elementType);
|
2018-06-23 06:52:02 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
/// Parse a dimension list of a tensor or memref type. This populates the
|
|
|
|
/// dimension list, returning -1 for the '?' dimensions.
|
|
|
|
///
|
|
|
|
/// dimension-list-ranked ::= (dimension `x`)*
|
|
|
|
/// dimension ::= `?` | integer-literal
|
|
|
|
///
|
|
|
|
ParseResult Parser::parseDimensionListRanked(SmallVectorImpl<int> &dimensions) {
|
2018-07-10 10:05:38 +08:00
|
|
|
while (getToken().isAny(Token::integer, Token::question)) {
|
2018-06-23 06:52:02 +08:00
|
|
|
if (consumeIf(Token::question)) {
|
|
|
|
dimensions.push_back(-1);
|
|
|
|
} else {
|
|
|
|
// Make sure this integer value is in bound and valid.
|
2018-07-10 10:05:38 +08:00
|
|
|
auto dimension = getToken().getUnsignedIntegerValue();
|
2018-06-23 06:52:02 +08:00
|
|
|
if (!dimension.hasValue() || (int)dimension.getValue() < 0)
|
|
|
|
return emitError("invalid dimension");
|
|
|
|
dimensions.push_back((int)dimension.getValue());
|
|
|
|
consumeToken(Token::integer);
|
|
|
|
}
|
|
|
|
|
|
|
|
// Make sure we have an 'x' or something like 'xbf32'.
|
2018-07-10 10:05:38 +08:00
|
|
|
if (getToken().isNot(Token::bare_identifier) ||
|
|
|
|
getTokenSpelling()[0] != 'x')
|
2018-06-23 06:52:02 +08:00
|
|
|
return emitError("expected 'x' in dimension list");
|
|
|
|
|
|
|
|
// If we had a prefix of 'x', lex the next token immediately after the 'x'.
|
2018-07-10 10:05:38 +08:00
|
|
|
if (getTokenSpelling().size() != 1)
|
|
|
|
state.lex.resetPointer(getTokenSpelling().data() + 1);
|
2018-06-23 06:52:02 +08:00
|
|
|
|
|
|
|
// Consume the 'x'.
|
|
|
|
consumeToken(Token::bare_identifier);
|
|
|
|
}
|
|
|
|
|
|
|
|
return ParseSuccess;
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Parse a tensor type.
|
|
|
|
///
|
|
|
|
/// tensor-type ::= `tensor` `<` dimension-list element-type `>`
|
|
|
|
/// dimension-list ::= dimension-list-ranked | `??`
|
|
|
|
///
|
2018-06-23 13:03:48 +08:00
|
|
|
Type *Parser::parseTensorType() {
|
2018-06-23 06:52:02 +08:00
|
|
|
consumeToken(Token::kw_tensor);
|
|
|
|
|
2018-07-24 08:30:01 +08:00
|
|
|
if (parseToken(Token::less, "expected '<' in tensor type"))
|
|
|
|
return nullptr;
|
2018-06-23 06:52:02 +08:00
|
|
|
|
|
|
|
bool isUnranked;
|
|
|
|
SmallVector<int, 4> dimensions;
|
|
|
|
|
|
|
|
if (consumeIf(Token::questionquestion)) {
|
|
|
|
isUnranked = true;
|
|
|
|
} else {
|
|
|
|
isUnranked = false;
|
|
|
|
if (parseDimensionListRanked(dimensions))
|
2018-06-23 13:03:48 +08:00
|
|
|
return nullptr;
|
2018-06-23 06:52:02 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
// Parse the element type.
|
Eliminate "primitive" types from being a thing, splitting them into FloatType
and OtherType. Other type is now the thing that holds AffineInt, Control,
eventually Resource, Variant, String, etc. FloatType holds the floating point
types, and allows convenient query of isa<FloatType>().
This fixes issues where we allowed control to be the element type of tensor,
memref, vector. At the same time, ban AffineInt from being an element of a
vector/memref/tensor as well since we don't need it.
I updated the spec to match this as well.
PiperOrigin-RevId: 206361942
2018-07-28 04:09:58 +08:00
|
|
|
auto typeLoc = getToken().getLoc();
|
|
|
|
auto *elementType = parseType();
|
|
|
|
if (!elementType || parseToken(Token::greater, "expected '>' in tensor type"))
|
2018-06-23 13:03:48 +08:00
|
|
|
return nullptr;
|
2018-06-23 06:52:02 +08:00
|
|
|
|
Eliminate "primitive" types from being a thing, splitting them into FloatType
and OtherType. Other type is now the thing that holds AffineInt, Control,
eventually Resource, Variant, String, etc. FloatType holds the floating point
types, and allows convenient query of isa<FloatType>().
This fixes issues where we allowed control to be the element type of tensor,
memref, vector. At the same time, ban AffineInt from being an element of a
vector/memref/tensor as well since we don't need it.
I updated the spec to match this as well.
PiperOrigin-RevId: 206361942
2018-07-28 04:09:58 +08:00
|
|
|
if (!isa<IntegerType>(elementType) && !isa<FloatType>(elementType) &&
|
|
|
|
!isa<VectorType>(elementType))
|
|
|
|
return (emitError(typeLoc, "invalid tensor element type"), nullptr);
|
2018-06-23 06:52:02 +08:00
|
|
|
|
2018-06-24 09:09:09 +08:00
|
|
|
if (isUnranked)
|
2018-07-09 11:51:38 +08:00
|
|
|
return builder.getTensorType(elementType);
|
|
|
|
return builder.getTensorType(dimensions, elementType);
|
2018-06-23 06:52:02 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
/// Parse a memref type.
|
|
|
|
///
|
|
|
|
/// memref-type ::= `memref` `<` dimension-list-ranked element-type
|
|
|
|
/// (`,` semi-affine-map-composition)? (`,` memory-space)? `>`
|
|
|
|
///
|
|
|
|
/// semi-affine-map-composition ::= (semi-affine-map `,` )* semi-affine-map
|
|
|
|
/// memory-space ::= integer-literal /* | TODO: address-space-id */
|
|
|
|
///
|
2018-06-23 13:03:48 +08:00
|
|
|
Type *Parser::parseMemRefType() {
|
2018-06-23 06:52:02 +08:00
|
|
|
consumeToken(Token::kw_memref);
|
|
|
|
|
2018-07-24 08:30:01 +08:00
|
|
|
if (parseToken(Token::less, "expected '<' in memref type"))
|
|
|
|
return nullptr;
|
2018-06-23 06:52:02 +08:00
|
|
|
|
|
|
|
SmallVector<int, 4> dimensions;
|
|
|
|
if (parseDimensionListRanked(dimensions))
|
2018-06-23 13:03:48 +08:00
|
|
|
return nullptr;
|
2018-06-23 06:52:02 +08:00
|
|
|
|
|
|
|
// Parse the element type.
|
Eliminate "primitive" types from being a thing, splitting them into FloatType
and OtherType. Other type is now the thing that holds AffineInt, Control,
eventually Resource, Variant, String, etc. FloatType holds the floating point
types, and allows convenient query of isa<FloatType>().
This fixes issues where we allowed control to be the element type of tensor,
memref, vector. At the same time, ban AffineInt from being an element of a
vector/memref/tensor as well since we don't need it.
I updated the spec to match this as well.
PiperOrigin-RevId: 206361942
2018-07-28 04:09:58 +08:00
|
|
|
auto typeLoc = getToken().getLoc();
|
|
|
|
auto *elementType = parseType();
|
2018-06-23 13:03:48 +08:00
|
|
|
if (!elementType)
|
|
|
|
return nullptr;
|
2018-06-23 06:52:02 +08:00
|
|
|
|
Eliminate "primitive" types from being a thing, splitting them into FloatType
and OtherType. Other type is now the thing that holds AffineInt, Control,
eventually Resource, Variant, String, etc. FloatType holds the floating point
types, and allows convenient query of isa<FloatType>().
This fixes issues where we allowed control to be the element type of tensor,
memref, vector. At the same time, ban AffineInt from being an element of a
vector/memref/tensor as well since we don't need it.
I updated the spec to match this as well.
PiperOrigin-RevId: 206361942
2018-07-28 04:09:58 +08:00
|
|
|
if (!isa<IntegerType>(elementType) && !isa<FloatType>(elementType) &&
|
|
|
|
!isa<VectorType>(elementType))
|
|
|
|
return (emitError(typeLoc, "invalid memref element type"), nullptr);
|
|
|
|
|
2018-07-17 00:45:22 +08:00
|
|
|
// Parse semi-affine-map-composition.
|
2018-07-24 07:56:32 +08:00
|
|
|
SmallVector<AffineMap *, 2> affineMapComposition;
|
2018-07-26 03:55:50 +08:00
|
|
|
unsigned memorySpace = 0;
|
2018-07-17 00:45:22 +08:00
|
|
|
bool parsedMemorySpace = false;
|
2018-06-23 06:52:02 +08:00
|
|
|
|
2018-07-17 00:45:22 +08:00
|
|
|
auto parseElt = [&]() -> ParseResult {
|
|
|
|
if (getToken().is(Token::integer)) {
|
|
|
|
// Parse memory space.
|
|
|
|
if (parsedMemorySpace)
|
|
|
|
return emitError("multiple memory spaces specified in memref type");
|
|
|
|
auto v = getToken().getUnsignedIntegerValue();
|
|
|
|
if (!v.hasValue())
|
|
|
|
return emitError("invalid memory space in memref type");
|
|
|
|
memorySpace = v.getValue();
|
|
|
|
consumeToken(Token::integer);
|
|
|
|
parsedMemorySpace = true;
|
|
|
|
} else {
|
|
|
|
// Parse affine map.
|
|
|
|
if (parsedMemorySpace)
|
|
|
|
return emitError("affine map after memory space in memref type");
|
2018-07-24 07:56:32 +08:00
|
|
|
auto *affineMap = parseAffineMapReference();
|
2018-07-17 00:45:22 +08:00
|
|
|
if (affineMap == nullptr)
|
|
|
|
return ParseFailure;
|
|
|
|
affineMapComposition.push_back(affineMap);
|
|
|
|
}
|
|
|
|
return ParseSuccess;
|
|
|
|
};
|
|
|
|
|
2018-07-26 03:55:50 +08:00
|
|
|
// Parse a list of mappings and address space if present.
|
|
|
|
if (consumeIf(Token::comma)) {
|
|
|
|
// Parse comma separated list of affine maps, followed by memory space.
|
|
|
|
if (parseCommaSeparatedListUntil(Token::greater, parseElt,
|
|
|
|
/*allowEmptyList=*/false)) {
|
|
|
|
return nullptr;
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
if (parseToken(Token::greater, "expected ',' or '>' in memref type"))
|
|
|
|
return nullptr;
|
2018-07-17 00:45:22 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
return MemRefType::get(dimensions, elementType, affineMapComposition,
|
|
|
|
memorySpace);
|
2018-06-23 06:52:02 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
/// Parse a function type.
|
|
|
|
///
|
|
|
|
/// function-type ::= type-list-parens `->` type-list
|
|
|
|
///
|
2018-06-23 13:03:48 +08:00
|
|
|
Type *Parser::parseFunctionType() {
|
2018-07-10 10:05:38 +08:00
|
|
|
assert(getToken().is(Token::l_paren));
|
2018-06-23 06:52:02 +08:00
|
|
|
|
2018-07-24 08:30:01 +08:00
|
|
|
SmallVector<Type *, 4> arguments, results;
|
|
|
|
if (parseTypeList(arguments) ||
|
|
|
|
parseToken(Token::arrow, "expected '->' in function type") ||
|
|
|
|
parseTypeList(results))
|
2018-06-23 13:03:48 +08:00
|
|
|
return nullptr;
|
2018-06-23 06:52:02 +08:00
|
|
|
|
2018-07-09 11:51:38 +08:00
|
|
|
return builder.getFunctionType(arguments, results);
|
2018-06-23 06:52:02 +08:00
|
|
|
}
|
|
|
|
|
2018-07-23 23:42:19 +08:00
|
|
|
/// Parse a list of types without an enclosing parenthesis. The list must have
|
|
|
|
/// at least one member.
|
|
|
|
///
|
|
|
|
/// type-list-no-parens ::= type (`,` type)*
|
|
|
|
///
|
|
|
|
ParseResult Parser::parseTypeListNoParens(SmallVectorImpl<Type *> &elements) {
|
|
|
|
auto parseElt = [&]() -> ParseResult {
|
|
|
|
auto elt = parseType();
|
|
|
|
elements.push_back(elt);
|
|
|
|
return elt ? ParseSuccess : ParseFailure;
|
|
|
|
};
|
|
|
|
|
|
|
|
return parseCommaSeparatedList(parseElt);
|
|
|
|
}
|
|
|
|
|
2018-06-23 06:52:02 +08:00
|
|
|
/// Parse a "type list", which is a singular type, or a parenthesized list of
|
|
|
|
/// types.
|
|
|
|
///
|
|
|
|
/// type-list ::= type-list-parens | type
|
|
|
|
/// type-list-parens ::= `(` `)`
|
2018-07-23 23:42:19 +08:00
|
|
|
/// | `(` type-list-no-parens `)`
|
2018-06-23 06:52:02 +08:00
|
|
|
///
|
2018-07-24 07:56:32 +08:00
|
|
|
ParseResult Parser::parseTypeList(SmallVectorImpl<Type *> &elements) {
|
2018-06-23 13:03:48 +08:00
|
|
|
auto parseElt = [&]() -> ParseResult {
|
|
|
|
auto elt = parseType();
|
|
|
|
elements.push_back(elt);
|
|
|
|
return elt ? ParseSuccess : ParseFailure;
|
|
|
|
};
|
|
|
|
|
2018-06-23 06:52:02 +08:00
|
|
|
// If there is no parens, then it must be a singular type.
|
|
|
|
if (!consumeIf(Token::l_paren))
|
2018-06-23 13:03:48 +08:00
|
|
|
return parseElt();
|
2018-06-23 06:52:02 +08:00
|
|
|
|
2018-07-22 05:32:09 +08:00
|
|
|
if (parseCommaSeparatedListUntil(Token::r_paren, parseElt))
|
2018-06-23 06:52:02 +08:00
|
|
|
return ParseFailure;
|
|
|
|
|
|
|
|
return ParseSuccess;
|
|
|
|
}
|
|
|
|
|
2018-07-05 11:45:39 +08:00
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
// Attribute parsing.
|
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
|
2018-08-22 08:55:22 +08:00
|
|
|
/// Given a parsed reference to a function name like @foo and a type that it
|
|
|
|
/// corresponds to, resolve it to a concrete function object (possibly
|
|
|
|
/// synthesizing a forward reference) or emit an error and return null on
|
|
|
|
/// failure.
|
|
|
|
Function *Parser::resolveFunctionReference(StringRef nameStr, SMLoc nameLoc,
|
|
|
|
FunctionType *type) {
|
|
|
|
Identifier name = builder.getIdentifier(nameStr.drop_front());
|
|
|
|
|
|
|
|
// See if the function has already been defined in the module.
|
|
|
|
Function *function = getModule()->getNamedFunction(name);
|
|
|
|
|
|
|
|
// If not, get or create a forward reference to one.
|
|
|
|
if (!function) {
|
|
|
|
auto &entry = state.functionForwardRefs[name];
|
|
|
|
if (!entry.first) {
|
|
|
|
entry.first = new ExtFunction(name, type);
|
|
|
|
entry.second = nameLoc;
|
|
|
|
}
|
|
|
|
function = entry.first;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (function->getType() != type)
|
|
|
|
return (emitError(nameLoc, "reference to function with mismatched type"),
|
|
|
|
nullptr);
|
|
|
|
return function;
|
|
|
|
}
|
|
|
|
|
2018-07-05 11:45:39 +08:00
|
|
|
/// Attribute parsing.
|
|
|
|
///
|
|
|
|
/// attribute-value ::= bool-literal
|
|
|
|
/// | integer-literal
|
|
|
|
/// | float-literal
|
|
|
|
/// | string-literal
|
2018-08-03 16:54:46 +08:00
|
|
|
/// | type
|
2018-07-05 11:45:39 +08:00
|
|
|
/// | `[` (attribute-value (`,` attribute-value)*)? `]`
|
2018-08-20 12:17:22 +08:00
|
|
|
/// | function-id `:` function-type
|
2018-07-05 11:45:39 +08:00
|
|
|
///
|
|
|
|
Attribute *Parser::parseAttribute() {
|
2018-07-10 10:05:38 +08:00
|
|
|
switch (getToken().getKind()) {
|
2018-07-05 11:45:39 +08:00
|
|
|
case Token::kw_true:
|
|
|
|
consumeToken(Token::kw_true);
|
2018-07-11 01:59:53 +08:00
|
|
|
return builder.getBoolAttr(true);
|
2018-07-05 11:45:39 +08:00
|
|
|
case Token::kw_false:
|
|
|
|
consumeToken(Token::kw_false);
|
2018-07-11 01:59:53 +08:00
|
|
|
return builder.getBoolAttr(false);
|
2018-07-05 11:45:39 +08:00
|
|
|
|
2018-08-01 08:15:15 +08:00
|
|
|
case Token::floatliteral: {
|
|
|
|
auto val = getToken().getFloatingPointValue();
|
|
|
|
if (!val.hasValue())
|
|
|
|
return (emitError("floating point value too large for attribute"),
|
|
|
|
nullptr);
|
|
|
|
consumeToken(Token::floatliteral);
|
|
|
|
return builder.getFloatAttr(val.getValue());
|
|
|
|
}
|
2018-07-05 11:45:39 +08:00
|
|
|
case Token::integer: {
|
2018-07-10 10:05:38 +08:00
|
|
|
auto val = getToken().getUInt64IntegerValue();
|
2018-07-05 11:45:39 +08:00
|
|
|
if (!val.hasValue() || (int64_t)val.getValue() < 0)
|
|
|
|
return (emitError("integer too large for attribute"), nullptr);
|
|
|
|
consumeToken(Token::integer);
|
2018-07-11 01:59:53 +08:00
|
|
|
return builder.getIntegerAttr((int64_t)val.getValue());
|
2018-07-05 11:45:39 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
case Token::minus: {
|
|
|
|
consumeToken(Token::minus);
|
2018-07-10 10:05:38 +08:00
|
|
|
if (getToken().is(Token::integer)) {
|
|
|
|
auto val = getToken().getUInt64IntegerValue();
|
2018-07-05 11:45:39 +08:00
|
|
|
if (!val.hasValue() || (int64_t)-val.getValue() >= 0)
|
|
|
|
return (emitError("integer too large for attribute"), nullptr);
|
|
|
|
consumeToken(Token::integer);
|
2018-07-11 01:59:53 +08:00
|
|
|
return builder.getIntegerAttr((int64_t)-val.getValue());
|
2018-07-05 11:45:39 +08:00
|
|
|
}
|
2018-08-01 08:15:15 +08:00
|
|
|
if (getToken().is(Token::floatliteral)) {
|
|
|
|
auto val = getToken().getFloatingPointValue();
|
|
|
|
if (!val.hasValue())
|
|
|
|
return (emitError("floating point value too large for attribute"),
|
|
|
|
nullptr);
|
|
|
|
consumeToken(Token::floatliteral);
|
|
|
|
return builder.getFloatAttr(-val.getValue());
|
|
|
|
}
|
2018-07-05 11:45:39 +08:00
|
|
|
|
|
|
|
return (emitError("expected constant integer or floating point value"),
|
|
|
|
nullptr);
|
|
|
|
}
|
|
|
|
|
|
|
|
case Token::string: {
|
2018-07-10 10:05:38 +08:00
|
|
|
auto val = getToken().getStringValue();
|
2018-07-05 11:45:39 +08:00
|
|
|
consumeToken(Token::string);
|
2018-07-11 01:59:53 +08:00
|
|
|
return builder.getStringAttr(val);
|
2018-07-05 11:45:39 +08:00
|
|
|
}
|
|
|
|
|
2018-07-26 02:15:20 +08:00
|
|
|
case Token::l_square: {
|
|
|
|
consumeToken(Token::l_square);
|
2018-07-24 07:56:32 +08:00
|
|
|
SmallVector<Attribute *, 4> elements;
|
2018-07-05 11:45:39 +08:00
|
|
|
|
|
|
|
auto parseElt = [&]() -> ParseResult {
|
|
|
|
elements.push_back(parseAttribute());
|
|
|
|
return elements.back() ? ParseSuccess : ParseFailure;
|
|
|
|
};
|
|
|
|
|
2018-07-26 02:15:20 +08:00
|
|
|
if (parseCommaSeparatedListUntil(Token::r_square, parseElt))
|
2018-07-05 11:45:39 +08:00
|
|
|
return nullptr;
|
2018-07-11 01:59:53 +08:00
|
|
|
return builder.getArrayAttr(elements);
|
2018-07-05 11:45:39 +08:00
|
|
|
}
|
2018-08-03 16:54:46 +08:00
|
|
|
case Token::hash_identifier:
|
|
|
|
case Token::l_paren: {
|
2018-07-19 07:29:21 +08:00
|
|
|
// Try to parse affine map reference.
|
2018-08-03 16:54:46 +08:00
|
|
|
if (auto *affineMap = parseAffineMapReference())
|
2018-07-19 07:29:21 +08:00
|
|
|
return builder.getAffineMapAttr(affineMap);
|
2018-07-05 11:45:39 +08:00
|
|
|
return (emitError("expected constant attribute value"), nullptr);
|
|
|
|
}
|
2018-08-20 12:17:22 +08:00
|
|
|
|
|
|
|
case Token::at_identifier: {
|
|
|
|
auto nameLoc = getToken().getLoc();
|
2018-08-22 08:55:22 +08:00
|
|
|
auto nameStr = getTokenSpelling();
|
2018-08-20 12:17:22 +08:00
|
|
|
consumeToken(Token::at_identifier);
|
|
|
|
|
|
|
|
if (parseToken(Token::colon, "expected ':' and function type"))
|
|
|
|
return nullptr;
|
|
|
|
auto typeLoc = getToken().getLoc();
|
|
|
|
Type *type = parseType();
|
|
|
|
if (!type)
|
|
|
|
return nullptr;
|
2018-08-22 08:55:22 +08:00
|
|
|
auto *fnType = dyn_cast<FunctionType>(type);
|
2018-08-20 12:17:22 +08:00
|
|
|
if (!fnType)
|
|
|
|
return (emitError(typeLoc, "expected function type"), nullptr);
|
|
|
|
|
2018-08-22 08:55:22 +08:00
|
|
|
auto *function = resolveFunctionReference(nameStr, nameLoc, fnType);
|
|
|
|
return function ? builder.getFunctionAttr(function) : nullptr;
|
2018-08-20 12:17:22 +08:00
|
|
|
}
|
|
|
|
|
2018-08-03 16:54:46 +08:00
|
|
|
default: {
|
|
|
|
if (Type *type = parseType())
|
|
|
|
return builder.getTypeAttr(type);
|
|
|
|
return nullptr;
|
|
|
|
}
|
|
|
|
}
|
2018-07-05 11:45:39 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
/// Attribute dictionary.
|
|
|
|
///
|
2018-07-10 00:00:25 +08:00
|
|
|
/// attribute-dict ::= `{` `}`
|
|
|
|
/// | `{` attribute-entry (`,` attribute-entry)* `}`
|
|
|
|
/// attribute-entry ::= bare-id `:` attribute-value
|
2018-07-05 11:45:39 +08:00
|
|
|
///
|
2018-07-24 07:56:32 +08:00
|
|
|
ParseResult
|
|
|
|
Parser::parseAttributeDict(SmallVectorImpl<NamedAttribute> &attributes) {
|
2018-07-05 11:45:39 +08:00
|
|
|
consumeToken(Token::l_brace);
|
|
|
|
|
|
|
|
auto parseElt = [&]() -> ParseResult {
|
|
|
|
// We allow keywords as attribute names.
|
2018-07-10 10:05:38 +08:00
|
|
|
if (getToken().isNot(Token::bare_identifier, Token::inttype) &&
|
|
|
|
!getToken().isKeyword())
|
2018-07-05 11:45:39 +08:00
|
|
|
return emitError("expected attribute name");
|
2018-07-11 01:59:53 +08:00
|
|
|
auto nameId = builder.getIdentifier(getTokenSpelling());
|
2018-07-05 11:45:39 +08:00
|
|
|
consumeToken();
|
|
|
|
|
2018-07-24 08:30:01 +08:00
|
|
|
if (parseToken(Token::colon, "expected ':' in attribute list"))
|
|
|
|
return ParseFailure;
|
2018-07-05 11:45:39 +08:00
|
|
|
|
|
|
|
auto attr = parseAttribute();
|
2018-07-24 07:56:32 +08:00
|
|
|
if (!attr)
|
|
|
|
return ParseFailure;
|
2018-07-05 11:45:39 +08:00
|
|
|
|
|
|
|
attributes.push_back({nameId, attr});
|
|
|
|
return ParseSuccess;
|
|
|
|
};
|
|
|
|
|
2018-07-22 05:32:09 +08:00
|
|
|
if (parseCommaSeparatedListUntil(Token::r_brace, parseElt))
|
2018-07-05 11:45:39 +08:00
|
|
|
return ParseFailure;
|
|
|
|
|
|
|
|
return ParseSuccess;
|
|
|
|
}
|
|
|
|
|
2018-06-28 02:03:08 +08:00
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
// Polyhedral structures.
|
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
|
2018-07-11 01:08:27 +08:00
|
|
|
/// Lower precedence ops (all at the same precedence level). LNoOp is false in
|
|
|
|
/// the boolean sense.
|
|
|
|
enum AffineLowPrecOp {
|
|
|
|
/// Null value.
|
|
|
|
LNoOp,
|
|
|
|
Add,
|
|
|
|
Sub
|
|
|
|
};
|
2018-06-28 02:03:08 +08:00
|
|
|
|
2018-07-11 01:08:27 +08:00
|
|
|
/// Higher precedence ops - all at the same precedence level. HNoOp is false in
|
|
|
|
/// the boolean sense.
|
|
|
|
enum AffineHighPrecOp {
|
|
|
|
/// Null value.
|
|
|
|
HNoOp,
|
|
|
|
Mul,
|
|
|
|
FloorDiv,
|
|
|
|
CeilDiv,
|
|
|
|
Mod
|
|
|
|
};
|
2018-06-30 09:09:29 +08:00
|
|
|
|
2018-07-11 01:08:27 +08:00
|
|
|
namespace {
|
2018-08-08 05:24:38 +08:00
|
|
|
/// This is a specialized parser for affine structures (affine maps, affine
|
|
|
|
/// expressions, and integer sets), maintaining the state transient to their
|
|
|
|
/// bodies.
|
|
|
|
class AffineParser : public Parser {
|
2018-07-11 01:08:27 +08:00
|
|
|
public:
|
2018-08-08 05:24:38 +08:00
|
|
|
explicit AffineParser(ParserState &state) : Parser(state) {}
|
2018-07-05 11:45:39 +08:00
|
|
|
|
2018-07-11 01:08:27 +08:00
|
|
|
AffineMap *parseAffineMapInline();
|
2018-08-08 05:24:38 +08:00
|
|
|
IntegerSet *parseIntegerSetInline();
|
2018-07-05 11:45:39 +08:00
|
|
|
|
2018-07-11 01:08:27 +08:00
|
|
|
private:
|
|
|
|
// Binary affine op parsing.
|
|
|
|
AffineLowPrecOp consumeIfLowPrecOp();
|
|
|
|
AffineHighPrecOp consumeIfHighPrecOp();
|
2018-06-28 02:03:08 +08:00
|
|
|
|
2018-07-11 01:08:27 +08:00
|
|
|
// Identifier lists for polyhedral structures.
|
2018-07-26 03:55:50 +08:00
|
|
|
ParseResult parseDimIdList(unsigned &numDims);
|
|
|
|
ParseResult parseSymbolIdList(unsigned &numSymbols);
|
|
|
|
ParseResult parseIdentifierDefinition(AffineExpr *idExpr);
|
2018-07-11 01:08:27 +08:00
|
|
|
|
|
|
|
AffineExpr *parseAffineExpr();
|
|
|
|
AffineExpr *parseParentheticalExpr();
|
|
|
|
AffineExpr *parseNegateExpression(AffineExpr *lhs);
|
|
|
|
AffineExpr *parseIntegerExpr();
|
|
|
|
AffineExpr *parseBareIdExpr();
|
|
|
|
|
|
|
|
AffineExpr *getBinaryAffineOpExpr(AffineHighPrecOp op, AffineExpr *lhs,
|
2018-07-21 05:57:21 +08:00
|
|
|
AffineExpr *rhs, SMLoc opLoc);
|
2018-07-11 01:08:27 +08:00
|
|
|
AffineExpr *getBinaryAffineOpExpr(AffineLowPrecOp op, AffineExpr *lhs,
|
|
|
|
AffineExpr *rhs);
|
|
|
|
AffineExpr *parseAffineOperandExpr(AffineExpr *lhs);
|
|
|
|
AffineExpr *parseAffineLowPrecOpExpr(AffineExpr *llhs,
|
|
|
|
AffineLowPrecOp llhsOp);
|
|
|
|
AffineExpr *parseAffineHighPrecOpExpr(AffineExpr *llhs,
|
2018-07-21 05:57:21 +08:00
|
|
|
AffineHighPrecOp llhsOp,
|
|
|
|
SMLoc llhsOpLoc);
|
2018-08-08 05:24:38 +08:00
|
|
|
AffineExpr *parseAffineConstraint(bool *isEq);
|
2018-07-11 01:08:27 +08:00
|
|
|
|
|
|
|
private:
|
2018-07-26 03:55:50 +08:00
|
|
|
SmallVector<std::pair<StringRef, AffineExpr *>, 4> dimsAndSymbols;
|
2018-07-11 01:08:27 +08:00
|
|
|
};
|
|
|
|
} // end anonymous namespace
|
2018-07-04 11:16:08 +08:00
|
|
|
|
2018-07-21 05:57:21 +08:00
|
|
|
/// Create an affine binary high precedence op expression (mul's, div's, mod).
|
|
|
|
/// opLoc is the location of the op token to be used to report errors
|
|
|
|
/// for non-conforming expressions.
|
2018-08-08 05:24:38 +08:00
|
|
|
AffineExpr *AffineParser::getBinaryAffineOpExpr(AffineHighPrecOp op,
|
|
|
|
AffineExpr *lhs,
|
|
|
|
AffineExpr *rhs, SMLoc opLoc) {
|
2018-07-12 12:31:07 +08:00
|
|
|
// TODO: make the error location info accurate.
|
2018-07-04 11:16:08 +08:00
|
|
|
switch (op) {
|
|
|
|
case Mul:
|
2018-07-20 04:07:16 +08:00
|
|
|
if (!lhs->isSymbolicOrConstant() && !rhs->isSymbolicOrConstant()) {
|
2018-07-21 05:57:21 +08:00
|
|
|
emitError(opLoc, "non-affine expression: at least one of the multiply "
|
|
|
|
"operands has to be either a constant or symbolic");
|
2018-07-10 00:00:25 +08:00
|
|
|
return nullptr;
|
|
|
|
}
|
2018-07-11 01:59:53 +08:00
|
|
|
return builder.getMulExpr(lhs, rhs);
|
2018-07-04 11:16:08 +08:00
|
|
|
case FloorDiv:
|
2018-07-20 04:07:16 +08:00
|
|
|
if (!rhs->isSymbolicOrConstant()) {
|
2018-07-21 05:57:21 +08:00
|
|
|
emitError(opLoc, "non-affine expression: right operand of floordiv "
|
|
|
|
"has to be either a constant or symbolic");
|
2018-07-10 00:00:25 +08:00
|
|
|
return nullptr;
|
|
|
|
}
|
2018-07-11 01:59:53 +08:00
|
|
|
return builder.getFloorDivExpr(lhs, rhs);
|
2018-07-04 11:16:08 +08:00
|
|
|
case CeilDiv:
|
2018-07-20 04:07:16 +08:00
|
|
|
if (!rhs->isSymbolicOrConstant()) {
|
2018-07-21 05:57:21 +08:00
|
|
|
emitError(opLoc, "non-affine expression: right operand of ceildiv "
|
|
|
|
"has to be either a constant or symbolic");
|
2018-07-10 00:00:25 +08:00
|
|
|
return nullptr;
|
|
|
|
}
|
2018-07-11 01:59:53 +08:00
|
|
|
return builder.getCeilDivExpr(lhs, rhs);
|
2018-07-04 11:16:08 +08:00
|
|
|
case Mod:
|
2018-07-20 04:07:16 +08:00
|
|
|
if (!rhs->isSymbolicOrConstant()) {
|
2018-07-21 05:57:21 +08:00
|
|
|
emitError(opLoc, "non-affine expression: right operand of mod "
|
|
|
|
"has to be either a constant or symbolic");
|
2018-07-10 00:00:25 +08:00
|
|
|
return nullptr;
|
|
|
|
}
|
2018-07-11 01:59:53 +08:00
|
|
|
return builder.getModExpr(lhs, rhs);
|
2018-07-04 11:16:08 +08:00
|
|
|
case HNoOp:
|
|
|
|
llvm_unreachable("can't create affine expression for null high prec op");
|
|
|
|
return nullptr;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-07-10 00:00:25 +08:00
|
|
|
/// Create an affine binary low precedence op expression (add, sub).
|
2018-08-08 05:24:38 +08:00
|
|
|
AffineExpr *AffineParser::getBinaryAffineOpExpr(AffineLowPrecOp op,
|
|
|
|
AffineExpr *lhs,
|
|
|
|
AffineExpr *rhs) {
|
2018-07-04 11:16:08 +08:00
|
|
|
switch (op) {
|
|
|
|
case AffineLowPrecOp::Add:
|
2018-07-11 01:59:53 +08:00
|
|
|
return builder.getAddExpr(lhs, rhs);
|
2018-07-04 11:16:08 +08:00
|
|
|
case AffineLowPrecOp::Sub:
|
2018-07-20 05:08:50 +08:00
|
|
|
return builder.getAddExpr(
|
|
|
|
lhs, builder.getMulExpr(rhs, builder.getConstantExpr(-1)));
|
2018-07-04 11:16:08 +08:00
|
|
|
case AffineLowPrecOp::LNoOp:
|
|
|
|
llvm_unreachable("can't create affine expression for null low prec op");
|
|
|
|
return nullptr;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Consume this token if it is a lower precedence affine op (there are only two
|
2018-07-10 00:00:25 +08:00
|
|
|
/// precedence levels).
|
2018-08-08 05:24:38 +08:00
|
|
|
AffineLowPrecOp AffineParser::consumeIfLowPrecOp() {
|
2018-07-10 10:05:38 +08:00
|
|
|
switch (getToken().getKind()) {
|
2018-07-04 11:16:08 +08:00
|
|
|
case Token::plus:
|
|
|
|
consumeToken(Token::plus);
|
|
|
|
return AffineLowPrecOp::Add;
|
|
|
|
case Token::minus:
|
|
|
|
consumeToken(Token::minus);
|
|
|
|
return AffineLowPrecOp::Sub;
|
|
|
|
default:
|
|
|
|
return AffineLowPrecOp::LNoOp;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Consume this token if it is a higher precedence affine op (there are only
|
|
|
|
/// two precedence levels)
|
2018-08-08 05:24:38 +08:00
|
|
|
AffineHighPrecOp AffineParser::consumeIfHighPrecOp() {
|
2018-07-10 10:05:38 +08:00
|
|
|
switch (getToken().getKind()) {
|
2018-07-04 11:16:08 +08:00
|
|
|
case Token::star:
|
|
|
|
consumeToken(Token::star);
|
|
|
|
return Mul;
|
|
|
|
case Token::kw_floordiv:
|
|
|
|
consumeToken(Token::kw_floordiv);
|
|
|
|
return FloorDiv;
|
|
|
|
case Token::kw_ceildiv:
|
|
|
|
consumeToken(Token::kw_ceildiv);
|
|
|
|
return CeilDiv;
|
|
|
|
case Token::kw_mod:
|
|
|
|
consumeToken(Token::kw_mod);
|
|
|
|
return Mod;
|
|
|
|
default:
|
|
|
|
return HNoOp;
|
|
|
|
}
|
2018-06-30 09:09:29 +08:00
|
|
|
}
|
|
|
|
|
2018-07-10 00:00:25 +08:00
|
|
|
/// Parse a high precedence op expression list: mul, div, and mod are high
|
|
|
|
/// precedence binary ops, i.e., parse a
|
|
|
|
/// expr_1 op_1 expr_2 op_2 ... expr_n
|
|
|
|
/// where op_1, op_2 are all a AffineHighPrecOp (mul, div, mod).
|
|
|
|
/// All affine binary ops are left associative.
|
|
|
|
/// Given llhs, returns (llhs llhsOp lhs) op rhs, or (lhs op rhs) if llhs is
|
|
|
|
/// null. If no rhs can be found, returns (llhs llhsOp lhs) or lhs if llhs is
|
2018-07-21 05:57:21 +08:00
|
|
|
/// null. llhsOpLoc is the location of the llhsOp token that will be used to
|
|
|
|
/// report an error for non-conforming expressions.
|
2018-08-08 05:24:38 +08:00
|
|
|
AffineExpr *AffineParser::parseAffineHighPrecOpExpr(AffineExpr *llhs,
|
|
|
|
AffineHighPrecOp llhsOp,
|
|
|
|
SMLoc llhsOpLoc) {
|
2018-07-11 01:08:27 +08:00
|
|
|
AffineExpr *lhs = parseAffineOperandExpr(llhs);
|
2018-07-10 00:00:25 +08:00
|
|
|
if (!lhs)
|
|
|
|
return nullptr;
|
|
|
|
|
|
|
|
// Found an LHS. Parse the remaining expression.
|
2018-07-21 05:57:21 +08:00
|
|
|
auto opLoc = getToken().getLoc();
|
2018-07-11 01:08:27 +08:00
|
|
|
if (AffineHighPrecOp op = consumeIfHighPrecOp()) {
|
2018-07-10 00:00:25 +08:00
|
|
|
if (llhs) {
|
2018-07-21 05:57:21 +08:00
|
|
|
AffineExpr *expr = getBinaryAffineOpExpr(llhsOp, llhs, lhs, opLoc);
|
2018-07-10 00:00:25 +08:00
|
|
|
if (!expr)
|
|
|
|
return nullptr;
|
2018-07-21 05:57:21 +08:00
|
|
|
return parseAffineHighPrecOpExpr(expr, op, opLoc);
|
2018-07-10 00:00:25 +08:00
|
|
|
}
|
|
|
|
// No LLHS, get RHS
|
2018-07-21 05:57:21 +08:00
|
|
|
return parseAffineHighPrecOpExpr(lhs, op, opLoc);
|
2018-07-10 00:00:25 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
// This is the last operand in this expression.
|
|
|
|
if (llhs)
|
2018-07-21 05:57:21 +08:00
|
|
|
return getBinaryAffineOpExpr(llhsOp, llhs, lhs, llhsOpLoc);
|
2018-07-10 00:00:25 +08:00
|
|
|
|
|
|
|
// No llhs, 'lhs' itself is the expression.
|
|
|
|
return lhs;
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Parse an affine expression inside parentheses.
|
|
|
|
///
|
|
|
|
/// affine-expr ::= `(` affine-expr `)`
|
2018-08-08 05:24:38 +08:00
|
|
|
AffineExpr *AffineParser::parseParentheticalExpr() {
|
2018-07-24 08:30:01 +08:00
|
|
|
if (parseToken(Token::l_paren, "expected '('"))
|
|
|
|
return nullptr;
|
2018-07-10 10:05:38 +08:00
|
|
|
if (getToken().is(Token::r_paren))
|
2018-07-10 00:00:25 +08:00
|
|
|
return (emitError("no expression inside parentheses"), nullptr);
|
2018-07-24 08:30:01 +08:00
|
|
|
|
2018-07-11 01:08:27 +08:00
|
|
|
auto *expr = parseAffineExpr();
|
2018-07-10 00:00:25 +08:00
|
|
|
if (!expr)
|
|
|
|
return nullptr;
|
2018-07-24 08:30:01 +08:00
|
|
|
if (parseToken(Token::r_paren, "expected ')'"))
|
|
|
|
return nullptr;
|
|
|
|
|
2018-07-10 00:00:25 +08:00
|
|
|
return expr;
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Parse the negation expression.
|
|
|
|
///
|
|
|
|
/// affine-expr ::= `-` affine-expr
|
2018-08-08 05:24:38 +08:00
|
|
|
AffineExpr *AffineParser::parseNegateExpression(AffineExpr *lhs) {
|
2018-07-24 08:30:01 +08:00
|
|
|
if (parseToken(Token::minus, "expected '-'"))
|
|
|
|
return nullptr;
|
2018-07-10 00:00:25 +08:00
|
|
|
|
2018-07-11 01:08:27 +08:00
|
|
|
AffineExpr *operand = parseAffineOperandExpr(lhs);
|
2018-07-10 00:00:25 +08:00
|
|
|
// Since negation has the highest precedence of all ops (including high
|
|
|
|
// precedence ops) but lower than parentheses, we are only going to use
|
|
|
|
// parseAffineOperandExpr instead of parseAffineExpr here.
|
|
|
|
if (!operand)
|
|
|
|
// Extra error message although parseAffineOperandExpr would have
|
|
|
|
// complained. Leads to a better diagnostic.
|
|
|
|
return (emitError("missing operand of negation"), nullptr);
|
2018-07-11 01:59:53 +08:00
|
|
|
auto *minusOne = builder.getConstantExpr(-1);
|
|
|
|
return builder.getMulExpr(minusOne, operand);
|
2018-07-10 00:00:25 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
/// Parse a bare id that may appear in an affine expression.
|
|
|
|
///
|
|
|
|
/// affine-expr ::= bare-id
|
2018-08-08 05:24:38 +08:00
|
|
|
AffineExpr *AffineParser::parseBareIdExpr() {
|
2018-07-10 10:05:38 +08:00
|
|
|
if (getToken().isNot(Token::bare_identifier))
|
2018-07-10 00:00:25 +08:00
|
|
|
return (emitError("expected bare identifier"), nullptr);
|
|
|
|
|
2018-07-10 10:05:38 +08:00
|
|
|
StringRef sRef = getTokenSpelling();
|
2018-07-26 03:55:50 +08:00
|
|
|
for (auto entry : dimsAndSymbols) {
|
2018-07-26 05:08:16 +08:00
|
|
|
if (entry.first == sRef) {
|
|
|
|
consumeToken(Token::bare_identifier);
|
|
|
|
return entry.second;
|
|
|
|
}
|
2018-07-10 00:00:25 +08:00
|
|
|
}
|
2018-07-12 12:31:07 +08:00
|
|
|
|
|
|
|
return (emitError("use of undeclared identifier"), nullptr);
|
2018-07-10 00:00:25 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
/// Parse a positive integral constant appearing in an affine expression.
|
|
|
|
///
|
|
|
|
/// affine-expr ::= integer-literal
|
2018-08-08 05:24:38 +08:00
|
|
|
AffineExpr *AffineParser::parseIntegerExpr() {
|
2018-07-10 10:05:38 +08:00
|
|
|
auto val = getToken().getUInt64IntegerValue();
|
2018-08-21 23:42:19 +08:00
|
|
|
if (!val.hasValue() || (int64_t)val.getValue() < 0)
|
2018-07-10 00:00:25 +08:00
|
|
|
return (emitError("constant too large for affineint"), nullptr);
|
2018-08-21 23:42:19 +08:00
|
|
|
|
2018-07-10 00:00:25 +08:00
|
|
|
consumeToken(Token::integer);
|
2018-07-11 01:59:53 +08:00
|
|
|
return builder.getConstantExpr((int64_t)val.getValue());
|
2018-07-10 00:00:25 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
/// Parses an expression that can be a valid operand of an affine expression.
|
2018-07-10 04:47:52 +08:00
|
|
|
/// lhs: if non-null, lhs is an affine expression that is the lhs of a binary
|
|
|
|
/// operator, the rhs of which is being parsed. This is used to determine
|
|
|
|
/// whether an error should be emitted for a missing right operand.
|
2018-07-10 00:00:25 +08:00
|
|
|
// Eg: for an expression without parentheses (like i + j + k + l), each
|
|
|
|
// of the four identifiers is an operand. For i + j*k + l, j*k is not an
|
|
|
|
// operand expression, it's an op expression and will be parsed via
|
|
|
|
// parseAffineHighPrecOpExpression(). However, for i + (j*k) + -l, (j*k) and -l
|
|
|
|
// are valid operands that will be parsed by this function.
|
2018-08-08 05:24:38 +08:00
|
|
|
AffineExpr *AffineParser::parseAffineOperandExpr(AffineExpr *lhs) {
|
2018-07-10 10:05:38 +08:00
|
|
|
switch (getToken().getKind()) {
|
2018-07-10 00:00:25 +08:00
|
|
|
case Token::bare_identifier:
|
2018-07-11 01:08:27 +08:00
|
|
|
return parseBareIdExpr();
|
2018-07-10 00:00:25 +08:00
|
|
|
case Token::integer:
|
2018-07-11 01:08:27 +08:00
|
|
|
return parseIntegerExpr();
|
2018-07-10 00:00:25 +08:00
|
|
|
case Token::l_paren:
|
2018-07-11 01:08:27 +08:00
|
|
|
return parseParentheticalExpr();
|
2018-07-10 00:00:25 +08:00
|
|
|
case Token::minus:
|
2018-07-11 01:08:27 +08:00
|
|
|
return parseNegateExpression(lhs);
|
2018-07-10 04:47:52 +08:00
|
|
|
case Token::kw_ceildiv:
|
|
|
|
case Token::kw_floordiv:
|
|
|
|
case Token::kw_mod:
|
|
|
|
case Token::plus:
|
|
|
|
case Token::star:
|
|
|
|
if (lhs)
|
|
|
|
emitError("missing right operand of binary operator");
|
|
|
|
else
|
|
|
|
emitError("missing left operand of binary operator");
|
|
|
|
return nullptr;
|
2018-07-10 00:00:25 +08:00
|
|
|
default:
|
|
|
|
if (lhs)
|
2018-07-10 04:47:52 +08:00
|
|
|
emitError("missing right operand of binary operator");
|
2018-07-10 00:00:25 +08:00
|
|
|
else
|
|
|
|
emitError("expected affine expression");
|
|
|
|
return nullptr;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-07-04 11:16:08 +08:00
|
|
|
/// Parse affine expressions that are bare-id's, integer constants,
|
|
|
|
/// parenthetical affine expressions, and affine op expressions that are a
|
|
|
|
/// composition of those.
|
2018-06-30 09:09:29 +08:00
|
|
|
///
|
2018-07-04 11:16:08 +08:00
|
|
|
/// All binary op's associate from left to right.
|
|
|
|
///
|
|
|
|
/// {add, sub} have lower precedence than {mul, div, and mod}.
|
|
|
|
///
|
2018-07-10 04:47:52 +08:00
|
|
|
/// Add, sub'are themselves at the same precedence level. Mul, floordiv,
|
|
|
|
/// ceildiv, and mod are at the same higher precedence level. Negation has
|
|
|
|
/// higher precedence than any binary op.
|
2018-07-04 11:16:08 +08:00
|
|
|
///
|
|
|
|
/// llhs: the affine expression appearing on the left of the one being parsed.
|
2018-07-10 00:00:25 +08:00
|
|
|
/// This function will return ((llhs llhsOp lhs) op rhs) if llhs is non null,
|
|
|
|
/// and lhs op rhs otherwise; if there is no rhs, llhs llhsOp lhs is returned if
|
|
|
|
/// llhs is non-null; otherwise lhs is returned. This is to deal with left
|
2018-07-04 11:16:08 +08:00
|
|
|
/// associativity.
|
|
|
|
///
|
|
|
|
/// Eg: when the expression is e1 + e2*e3 + e4, with e1 as llhs, this function
|
2018-07-10 00:00:25 +08:00
|
|
|
/// will return the affine expr equivalent of (e1 + (e2*e3)) + e4, where (e2*e3)
|
|
|
|
/// will be parsed using parseAffineHighPrecOpExpr().
|
2018-08-08 05:24:38 +08:00
|
|
|
AffineExpr *AffineParser::parseAffineLowPrecOpExpr(AffineExpr *llhs,
|
|
|
|
AffineLowPrecOp llhsOp) {
|
2018-07-10 04:47:52 +08:00
|
|
|
AffineExpr *lhs;
|
2018-07-11 01:08:27 +08:00
|
|
|
if (!(lhs = parseAffineOperandExpr(llhs)))
|
2018-07-10 00:00:25 +08:00
|
|
|
return nullptr;
|
2018-07-04 11:16:08 +08:00
|
|
|
|
|
|
|
// Found an LHS. Deal with the ops.
|
2018-07-11 01:08:27 +08:00
|
|
|
if (AffineLowPrecOp lOp = consumeIfLowPrecOp()) {
|
2018-07-04 11:16:08 +08:00
|
|
|
if (llhs) {
|
2018-07-09 11:51:38 +08:00
|
|
|
AffineExpr *sum = getBinaryAffineOpExpr(llhsOp, llhs, lhs);
|
2018-07-11 01:08:27 +08:00
|
|
|
return parseAffineLowPrecOpExpr(sum, lOp);
|
2018-07-04 11:16:08 +08:00
|
|
|
}
|
|
|
|
// No LLHS, get RHS and form the expression.
|
2018-07-11 01:08:27 +08:00
|
|
|
return parseAffineLowPrecOpExpr(lhs, lOp);
|
2018-07-10 00:00:25 +08:00
|
|
|
}
|
2018-07-21 05:57:21 +08:00
|
|
|
auto opLoc = getToken().getLoc();
|
2018-07-11 01:08:27 +08:00
|
|
|
if (AffineHighPrecOp hOp = consumeIfHighPrecOp()) {
|
2018-07-04 11:16:08 +08:00
|
|
|
// We have a higher precedence op here. Get the rhs operand for the llhs
|
|
|
|
// through parseAffineHighPrecOpExpr.
|
2018-07-21 05:57:21 +08:00
|
|
|
AffineExpr *highRes = parseAffineHighPrecOpExpr(lhs, hOp, opLoc);
|
2018-07-10 00:00:25 +08:00
|
|
|
if (!highRes)
|
|
|
|
return nullptr;
|
2018-07-11 01:08:27 +08:00
|
|
|
|
2018-07-04 11:16:08 +08:00
|
|
|
// If llhs is null, the product forms the first operand of the yet to be
|
2018-07-10 00:00:25 +08:00
|
|
|
// found expression. If non-null, the op to associate with llhs is llhsOp.
|
2018-07-04 11:16:08 +08:00
|
|
|
AffineExpr *expr =
|
2018-07-09 11:51:38 +08:00
|
|
|
llhs ? getBinaryAffineOpExpr(llhsOp, llhs, highRes) : highRes;
|
2018-07-11 01:08:27 +08:00
|
|
|
|
2018-07-10 00:00:25 +08:00
|
|
|
// Recurse for subsequent low prec op's after the affine high prec op
|
|
|
|
// expression.
|
2018-07-11 01:08:27 +08:00
|
|
|
if (AffineLowPrecOp nextOp = consumeIfLowPrecOp())
|
|
|
|
return parseAffineLowPrecOpExpr(expr, nextOp);
|
2018-07-04 11:16:08 +08:00
|
|
|
return expr;
|
|
|
|
}
|
2018-07-10 00:00:25 +08:00
|
|
|
// Last operand in the expression list.
|
|
|
|
if (llhs)
|
|
|
|
return getBinaryAffineOpExpr(llhsOp, llhs, lhs);
|
|
|
|
// No llhs, 'lhs' itself is the expression.
|
|
|
|
return lhs;
|
2018-07-04 11:16:08 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
/// Parse an affine expression.
|
2018-07-10 00:00:25 +08:00
|
|
|
/// affine-expr ::= `(` affine-expr `)`
|
|
|
|
/// | `-` affine-expr
|
|
|
|
/// | affine-expr `+` affine-expr
|
|
|
|
/// | affine-expr `-` affine-expr
|
|
|
|
/// | affine-expr `*` affine-expr
|
|
|
|
/// | affine-expr `floordiv` affine-expr
|
|
|
|
/// | affine-expr `ceildiv` affine-expr
|
|
|
|
/// | affine-expr `mod` affine-expr
|
|
|
|
/// | bare-id
|
|
|
|
/// | integer-literal
|
|
|
|
///
|
|
|
|
/// Additional conditions are checked depending on the production. For eg., one
|
|
|
|
/// of the operands for `*` has to be either constant/symbolic; the second
|
|
|
|
/// operand for floordiv, ceildiv, and mod has to be a positive integer.
|
2018-08-08 05:24:38 +08:00
|
|
|
AffineExpr *AffineParser::parseAffineExpr() {
|
2018-07-11 01:08:27 +08:00
|
|
|
return parseAffineLowPrecOpExpr(nullptr, AffineLowPrecOp::LNoOp);
|
2018-06-30 09:09:29 +08:00
|
|
|
}
|
|
|
|
|
2018-07-04 11:16:08 +08:00
|
|
|
/// Parse a dim or symbol from the lists appearing before the actual expressions
|
2018-07-11 01:08:27 +08:00
|
|
|
/// of the affine map. Update our state to store the dimensional/symbolic
|
2018-07-26 03:55:50 +08:00
|
|
|
/// identifier.
|
2018-08-08 05:24:38 +08:00
|
|
|
ParseResult AffineParser::parseIdentifierDefinition(AffineExpr *idExpr) {
|
2018-07-10 10:05:38 +08:00
|
|
|
if (getToken().isNot(Token::bare_identifier))
|
2018-07-04 11:16:08 +08:00
|
|
|
return emitError("expected bare identifier");
|
2018-07-26 03:55:50 +08:00
|
|
|
|
|
|
|
auto name = getTokenSpelling();
|
|
|
|
for (auto entry : dimsAndSymbols) {
|
|
|
|
if (entry.first == name)
|
|
|
|
return emitError("redefinition of identifier '" + Twine(name) + "'");
|
|
|
|
}
|
2018-06-30 09:09:29 +08:00
|
|
|
consumeToken(Token::bare_identifier);
|
2018-07-26 03:55:50 +08:00
|
|
|
|
|
|
|
dimsAndSymbols.push_back({name, idExpr});
|
2018-07-04 11:16:08 +08:00
|
|
|
return ParseSuccess;
|
2018-06-30 09:09:29 +08:00
|
|
|
}
|
|
|
|
|
2018-07-04 11:16:08 +08:00
|
|
|
/// Parse the list of symbolic identifiers to an affine map.
|
2018-08-08 05:24:38 +08:00
|
|
|
ParseResult AffineParser::parseSymbolIdList(unsigned &numSymbols) {
|
2018-07-26 03:55:50 +08:00
|
|
|
consumeToken(Token::l_square);
|
|
|
|
auto parseElt = [&]() -> ParseResult {
|
|
|
|
auto *symbol = AffineSymbolExpr::get(numSymbols++, getContext());
|
|
|
|
return parseIdentifierDefinition(symbol);
|
|
|
|
};
|
2018-07-26 02:15:20 +08:00
|
|
|
return parseCommaSeparatedListUntil(Token::r_square, parseElt);
|
2018-06-30 09:09:29 +08:00
|
|
|
}
|
|
|
|
|
2018-07-04 11:16:08 +08:00
|
|
|
/// Parse the list of dimensional identifiers to an affine map.
|
2018-08-08 05:24:38 +08:00
|
|
|
ParseResult AffineParser::parseDimIdList(unsigned &numDims) {
|
2018-07-24 08:30:01 +08:00
|
|
|
if (parseToken(Token::l_paren,
|
|
|
|
"expected '(' at start of dimensional identifiers list"))
|
|
|
|
return ParseFailure;
|
2018-06-30 09:09:29 +08:00
|
|
|
|
2018-07-26 03:55:50 +08:00
|
|
|
auto parseElt = [&]() -> ParseResult {
|
|
|
|
auto *dimension = AffineDimExpr::get(numDims++, getContext());
|
|
|
|
return parseIdentifierDefinition(dimension);
|
|
|
|
};
|
2018-07-22 05:32:09 +08:00
|
|
|
return parseCommaSeparatedListUntil(Token::r_paren, parseElt);
|
2018-06-30 09:09:29 +08:00
|
|
|
}
|
|
|
|
|
2018-07-04 11:16:08 +08:00
|
|
|
/// Parse an affine map definition.
|
2018-06-30 09:09:29 +08:00
|
|
|
///
|
2018-07-10 00:00:25 +08:00
|
|
|
/// affine-map-inline ::= dim-and-symbol-id-lists `->` multi-dim-affine-expr
|
|
|
|
/// (`size` `(` dim-size (`,` dim-size)* `)`)?
|
|
|
|
/// dim-size ::= affine-expr | `min` `(` affine-expr ( `,` affine-expr)+ `)`
|
2018-06-30 09:09:29 +08:00
|
|
|
///
|
2018-07-10 00:00:25 +08:00
|
|
|
/// multi-dim-affine-expr ::= `(` affine-expr (`,` affine-expr)* `)
|
2018-08-08 05:24:38 +08:00
|
|
|
AffineMap *AffineParser::parseAffineMapInline() {
|
2018-07-26 03:55:50 +08:00
|
|
|
unsigned numDims = 0, numSymbols = 0;
|
|
|
|
|
2018-06-30 09:09:29 +08:00
|
|
|
// List of dimensional identifiers.
|
2018-07-26 03:55:50 +08:00
|
|
|
if (parseDimIdList(numDims))
|
2018-07-05 11:45:39 +08:00
|
|
|
return nullptr;
|
2018-06-30 09:09:29 +08:00
|
|
|
|
|
|
|
// Symbols are optional.
|
2018-07-26 02:15:20 +08:00
|
|
|
if (getToken().is(Token::l_square)) {
|
2018-07-26 03:55:50 +08:00
|
|
|
if (parseSymbolIdList(numSymbols))
|
2018-07-05 11:45:39 +08:00
|
|
|
return nullptr;
|
2018-06-30 09:09:29 +08:00
|
|
|
}
|
2018-07-24 08:30:01 +08:00
|
|
|
|
|
|
|
if (parseToken(Token::arrow, "expected '->' or '['") ||
|
|
|
|
parseToken(Token::l_paren, "expected '(' at start of affine map range"))
|
2018-07-05 11:45:39 +08:00
|
|
|
return nullptr;
|
2018-06-30 09:09:29 +08:00
|
|
|
|
|
|
|
SmallVector<AffineExpr *, 4> exprs;
|
|
|
|
auto parseElt = [&]() -> ParseResult {
|
2018-07-11 01:08:27 +08:00
|
|
|
auto *elt = parseAffineExpr();
|
2018-06-30 09:09:29 +08:00
|
|
|
ParseResult res = elt ? ParseSuccess : ParseFailure;
|
|
|
|
exprs.push_back(elt);
|
|
|
|
return res;
|
|
|
|
};
|
|
|
|
|
|
|
|
// Parse a multi-dimensional affine expression (a comma-separated list of 1-d
|
2018-07-04 11:16:08 +08:00
|
|
|
// affine expressions); the list cannot be empty.
|
|
|
|
// Grammar: multi-dim-affine-expr ::= `(` affine-expr (`,` affine-expr)* `)
|
2018-07-22 05:32:09 +08:00
|
|
|
if (parseCommaSeparatedListUntil(Token::r_paren, parseElt, false))
|
2018-07-05 11:45:39 +08:00
|
|
|
return nullptr;
|
2018-06-30 09:09:29 +08:00
|
|
|
|
2018-07-12 12:31:07 +08:00
|
|
|
// Parse optional range sizes.
|
2018-07-13 09:04:04 +08:00
|
|
|
// range-sizes ::= (`size` `(` dim-size (`,` dim-size)* `)`)?
|
|
|
|
// dim-size ::= affine-expr | `min` `(` affine-expr (`,` affine-expr)+ `)`
|
|
|
|
// TODO(bondhugula): support for min of several affine expressions.
|
2018-07-12 12:31:07 +08:00
|
|
|
// TODO: check if sizes are non-negative whenever they are constant.
|
|
|
|
SmallVector<AffineExpr *, 4> rangeSizes;
|
|
|
|
if (consumeIf(Token::kw_size)) {
|
|
|
|
// Location of the l_paren token (if it exists) for error reporting later.
|
|
|
|
auto loc = getToken().getLoc();
|
2018-07-24 08:30:01 +08:00
|
|
|
if (parseToken(Token::l_paren, "expected '(' at start of affine map range"))
|
|
|
|
return nullptr;
|
2018-07-12 12:31:07 +08:00
|
|
|
|
|
|
|
auto parseRangeSize = [&]() -> ParseResult {
|
2018-07-26 03:55:50 +08:00
|
|
|
auto loc = getToken().getLoc();
|
2018-07-12 12:31:07 +08:00
|
|
|
auto *elt = parseAffineExpr();
|
2018-07-26 03:55:50 +08:00
|
|
|
if (!elt)
|
|
|
|
return ParseFailure;
|
|
|
|
|
|
|
|
if (!elt->isSymbolicOrConstant())
|
|
|
|
return emitError(loc,
|
|
|
|
"size expressions cannot refer to dimension values");
|
|
|
|
|
2018-07-12 12:31:07 +08:00
|
|
|
rangeSizes.push_back(elt);
|
2018-07-26 03:55:50 +08:00
|
|
|
return ParseSuccess;
|
2018-07-12 12:31:07 +08:00
|
|
|
};
|
|
|
|
|
2018-07-22 05:32:09 +08:00
|
|
|
if (parseCommaSeparatedListUntil(Token::r_paren, parseRangeSize, false))
|
2018-07-12 12:31:07 +08:00
|
|
|
return nullptr;
|
|
|
|
if (exprs.size() > rangeSizes.size())
|
|
|
|
return (emitError(loc, "fewer range sizes than range expressions"),
|
|
|
|
nullptr);
|
|
|
|
if (exprs.size() < rangeSizes.size())
|
|
|
|
return (emitError(loc, "more range sizes than range expressions"),
|
|
|
|
nullptr);
|
|
|
|
}
|
|
|
|
|
2018-07-04 11:16:08 +08:00
|
|
|
// Parsed a valid affine map.
|
2018-07-26 03:55:50 +08:00
|
|
|
return builder.getAffineMap(numDims, numSymbols, exprs, rangeSizes);
|
2018-06-28 02:03:08 +08:00
|
|
|
}
|
|
|
|
|
2018-07-11 01:08:27 +08:00
|
|
|
AffineMap *Parser::parseAffineMapInline() {
|
2018-08-08 05:24:38 +08:00
|
|
|
return AffineParser(state).parseAffineMapInline();
|
2018-07-11 01:08:27 +08:00
|
|
|
}
|
|
|
|
|
2018-07-17 00:45:22 +08:00
|
|
|
AffineMap *Parser::parseAffineMapReference() {
|
|
|
|
if (getToken().is(Token::hash_identifier)) {
|
|
|
|
// Parse affine map identifier and verify that it exists.
|
|
|
|
StringRef affineMapId = getTokenSpelling().drop_front();
|
|
|
|
if (getState().affineMapDefinitions.count(affineMapId) == 0)
|
|
|
|
return (emitError("undefined affine map id '" + affineMapId + "'"),
|
|
|
|
nullptr);
|
|
|
|
consumeToken(Token::hash_identifier);
|
|
|
|
return getState().affineMapDefinitions[affineMapId];
|
|
|
|
}
|
|
|
|
// Try to parse inline affine map.
|
|
|
|
return parseAffineMapInline();
|
|
|
|
}
|
|
|
|
|
2018-06-23 01:39:19 +08:00
|
|
|
//===----------------------------------------------------------------------===//
|
2018-07-19 23:35:28 +08:00
|
|
|
// FunctionParser
|
2018-06-23 01:39:19 +08:00
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
|
2018-07-19 23:35:28 +08:00
|
|
|
namespace {
|
|
|
|
/// This class contains parser state that is common across CFG and ML functions,
|
|
|
|
/// notably for dealing with operations and SSA values.
|
|
|
|
class FunctionParser : public Parser {
|
|
|
|
public:
|
2018-07-27 09:09:20 +08:00
|
|
|
enum class Kind { CFGFunc, MLFunc };
|
|
|
|
|
|
|
|
Kind getKind() const { return kind; }
|
2018-07-19 23:35:28 +08:00
|
|
|
|
2018-07-21 09:41:34 +08:00
|
|
|
/// After the function is finished parsing, this function checks to see if
|
|
|
|
/// there are any remaining issues.
|
2018-07-22 05:32:09 +08:00
|
|
|
ParseResult finalizeFunction(Function *func, SMLoc loc);
|
2018-07-21 09:41:34 +08:00
|
|
|
|
|
|
|
/// This represents a use of an SSA value in the program. The first two
|
|
|
|
/// entries in the tuple are the name and result number of a reference. The
|
|
|
|
/// third is the location of the reference, which is used in case this ends up
|
|
|
|
/// being a use of an undefined value.
|
|
|
|
struct SSAUseInfo {
|
|
|
|
StringRef name; // Value name, e.g. %42 or %abc
|
|
|
|
unsigned number; // Number, specified with #12
|
|
|
|
SMLoc loc; // Location of first definition or use.
|
|
|
|
};
|
2018-07-19 23:35:28 +08:00
|
|
|
|
|
|
|
/// Given a reference to an SSA value and its type, return a reference. This
|
|
|
|
/// returns null on failure.
|
|
|
|
SSAValue *resolveSSAUse(SSAUseInfo useInfo, Type *type);
|
|
|
|
|
|
|
|
/// Register a definition of a value with the symbol table.
|
|
|
|
ParseResult addDefinition(SSAUseInfo useInfo, SSAValue *value);
|
|
|
|
|
|
|
|
// SSA parsing productions.
|
|
|
|
ParseResult parseSSAUse(SSAUseInfo &result);
|
2018-07-22 05:32:09 +08:00
|
|
|
ParseResult parseOptionalSSAUseList(SmallVectorImpl<SSAUseInfo> &results);
|
2018-07-23 06:45:24 +08:00
|
|
|
|
|
|
|
template <typename ResultType>
|
|
|
|
ResultType parseSSADefOrUseAndType(
|
|
|
|
const std::function<ResultType(SSAUseInfo, Type *)> &action);
|
|
|
|
|
|
|
|
SSAValue *parseSSAUseAndType() {
|
|
|
|
return parseSSADefOrUseAndType<SSAValue *>(
|
|
|
|
[&](SSAUseInfo useInfo, Type *type) -> SSAValue * {
|
|
|
|
return resolveSSAUse(useInfo, type);
|
|
|
|
});
|
|
|
|
}
|
2018-07-22 05:32:09 +08:00
|
|
|
|
|
|
|
template <typename ValueTy>
|
2018-07-19 23:35:28 +08:00
|
|
|
ParseResult
|
2018-07-24 02:56:17 +08:00
|
|
|
parseOptionalSSAUseAndTypeList(SmallVectorImpl<ValueTy *> &results,
|
|
|
|
bool isParenthesized);
|
2018-07-19 23:35:28 +08:00
|
|
|
|
|
|
|
// Operations
|
|
|
|
ParseResult parseOperation(const CreateOperationFunction &createOpFunc);
|
2018-07-26 02:15:20 +08:00
|
|
|
Operation *parseVerboseOperation(const CreateOperationFunction &createOpFunc);
|
|
|
|
Operation *parseCustomOperation(const CreateOperationFunction &createOpFunc);
|
2018-07-19 23:35:28 +08:00
|
|
|
|
2018-07-27 09:09:20 +08:00
|
|
|
protected:
|
|
|
|
FunctionParser(ParserState &state, Kind kind) : Parser(state), kind(kind) {}
|
|
|
|
|
2018-07-19 23:35:28 +08:00
|
|
|
private:
|
2018-07-27 09:09:20 +08:00
|
|
|
/// Kind indicates if this is CFG or ML function parser.
|
|
|
|
Kind kind;
|
2018-07-19 23:35:28 +08:00
|
|
|
/// This keeps track of all of the SSA values we are tracking, indexed by
|
2018-07-21 09:41:34 +08:00
|
|
|
/// their name. This has one entry per result number.
|
|
|
|
llvm::StringMap<SmallVector<std::pair<SSAValue *, SMLoc>, 1>> values;
|
|
|
|
|
|
|
|
/// These are all of the placeholders we've made along with the location of
|
|
|
|
/// their first reference, to allow checking for use of undefined values.
|
|
|
|
DenseMap<SSAValue *, SMLoc> forwardReferencePlaceholders;
|
|
|
|
|
|
|
|
SSAValue *createForwardReferencePlaceholder(SMLoc loc, Type *type);
|
|
|
|
|
|
|
|
/// Return true if this is a forward reference.
|
|
|
|
bool isForwardReferencePlaceholder(SSAValue *value) {
|
|
|
|
return forwardReferencePlaceholders.count(value);
|
|
|
|
}
|
2018-07-19 23:35:28 +08:00
|
|
|
};
|
|
|
|
} // end anonymous namespace
|
|
|
|
|
2018-07-21 09:41:34 +08:00
|
|
|
/// Create and remember a new placeholder for a forward reference.
|
|
|
|
SSAValue *FunctionParser::createForwardReferencePlaceholder(SMLoc loc,
|
|
|
|
Type *type) {
|
|
|
|
// Forward references are always created as instructions, even in ML
|
|
|
|
// functions, because we just need something with a def/use chain.
|
|
|
|
//
|
|
|
|
// We create these placeholders as having an empty name, which we know cannot
|
|
|
|
// be created through normal user input, allowing us to distinguish them.
|
|
|
|
auto name = Identifier::get("placeholder", getContext());
|
2018-08-24 05:32:25 +08:00
|
|
|
auto *inst =
|
|
|
|
// FIXME(clattner): encode the location into the placeholder instead of
|
|
|
|
// into the forwardReferencePlaceholders map!
|
|
|
|
OperationInst::create(/*location=*/nullptr, name, /*operands=*/{}, type,
|
|
|
|
/*attrs=*/{}, getContext());
|
2018-07-21 09:41:34 +08:00
|
|
|
forwardReferencePlaceholders[inst->getResult(0)] = loc;
|
|
|
|
return inst->getResult(0);
|
|
|
|
}
|
|
|
|
|
2018-07-27 09:09:20 +08:00
|
|
|
/// Given an unbound reference to an SSA value and its type, return the value
|
2018-07-19 23:35:28 +08:00
|
|
|
/// it specifies. This returns null on failure.
|
|
|
|
SSAValue *FunctionParser::resolveSSAUse(SSAUseInfo useInfo, Type *type) {
|
2018-07-21 09:41:34 +08:00
|
|
|
auto &entries = values[useInfo.name];
|
|
|
|
|
2018-07-19 23:35:28 +08:00
|
|
|
// If we have already seen a value of this name, return it.
|
2018-07-21 09:41:34 +08:00
|
|
|
if (useInfo.number < entries.size() && entries[useInfo.number].first) {
|
|
|
|
auto *result = entries[useInfo.number].first;
|
2018-07-19 23:35:28 +08:00
|
|
|
// Check that the type matches the other uses.
|
|
|
|
if (result->getType() == type)
|
|
|
|
return result;
|
|
|
|
|
2018-07-21 09:41:34 +08:00
|
|
|
emitError(useInfo.loc, "use of value '" + useInfo.name.str() +
|
|
|
|
"' expects different type than prior uses");
|
|
|
|
emitError(entries[useInfo.number].second, "prior use here");
|
2018-07-19 23:35:28 +08:00
|
|
|
return nullptr;
|
|
|
|
}
|
|
|
|
|
2018-07-21 09:41:34 +08:00
|
|
|
// Make sure we have enough slots for this.
|
|
|
|
if (entries.size() <= useInfo.number)
|
|
|
|
entries.resize(useInfo.number + 1);
|
|
|
|
|
|
|
|
// If the value has already been defined and this is an overly large result
|
|
|
|
// number, diagnose that.
|
|
|
|
if (entries[0].first && !isForwardReferencePlaceholder(entries[0].first))
|
|
|
|
return (emitError(useInfo.loc, "reference to invalid result number"),
|
|
|
|
nullptr);
|
|
|
|
|
2018-08-01 14:14:16 +08:00
|
|
|
// Otherwise, this is a forward reference. If we are in ML function return
|
|
|
|
// an error. In CFG function, create a placeholder and remember
|
|
|
|
// that we did so.
|
2018-07-27 09:09:20 +08:00
|
|
|
if (getKind() == Kind::MLFunc)
|
|
|
|
return (
|
|
|
|
emitError(useInfo.loc, "use of undefined SSA value " + useInfo.name),
|
|
|
|
nullptr);
|
|
|
|
|
2018-07-21 09:41:34 +08:00
|
|
|
auto *result = createForwardReferencePlaceholder(useInfo.loc, type);
|
|
|
|
entries[useInfo.number].first = result;
|
|
|
|
entries[useInfo.number].second = useInfo.loc;
|
|
|
|
return result;
|
2018-07-19 23:35:28 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
/// Register a definition of a value with the symbol table.
|
|
|
|
ParseResult FunctionParser::addDefinition(SSAUseInfo useInfo, SSAValue *value) {
|
2018-07-21 09:41:34 +08:00
|
|
|
auto &entries = values[useInfo.name];
|
|
|
|
|
|
|
|
// Make sure there is a slot for this value.
|
|
|
|
if (entries.size() <= useInfo.number)
|
|
|
|
entries.resize(useInfo.number + 1);
|
|
|
|
|
|
|
|
// If we already have an entry for this, check to see if it was a definition
|
|
|
|
// or a forward reference.
|
|
|
|
if (auto *existing = entries[useInfo.number].first) {
|
|
|
|
if (!isForwardReferencePlaceholder(existing)) {
|
|
|
|
emitError(useInfo.loc,
|
|
|
|
"redefinition of SSA value '" + useInfo.name + "'");
|
|
|
|
return emitError(entries[useInfo.number].second,
|
|
|
|
"previously defined here");
|
|
|
|
}
|
2018-07-19 23:35:28 +08:00
|
|
|
|
2018-07-21 09:41:34 +08:00
|
|
|
// If it was a forward reference, update everything that used it to use the
|
|
|
|
// actual definition instead, delete the forward ref, and remove it from our
|
|
|
|
// set of forward references we track.
|
|
|
|
existing->replaceAllUsesWith(value);
|
|
|
|
existing->getDefiningInst()->destroy();
|
|
|
|
forwardReferencePlaceholders.erase(existing);
|
|
|
|
}
|
2018-07-19 23:35:28 +08:00
|
|
|
|
2018-07-21 09:41:34 +08:00
|
|
|
entries[useInfo.number].first = value;
|
|
|
|
entries[useInfo.number].second = useInfo.loc;
|
|
|
|
return ParseSuccess;
|
|
|
|
}
|
|
|
|
|
|
|
|
/// After the function is finished parsing, this function checks to see if
|
|
|
|
/// there are any remaining issues.
|
2018-07-22 05:32:09 +08:00
|
|
|
ParseResult FunctionParser::finalizeFunction(Function *func, SMLoc loc) {
|
2018-07-21 09:41:34 +08:00
|
|
|
// Check for any forward references that are left. If we find any, error out.
|
|
|
|
if (!forwardReferencePlaceholders.empty()) {
|
|
|
|
SmallVector<std::pair<const char *, SSAValue *>, 4> errors;
|
|
|
|
// Iteration over the map isn't determinstic, so sort by source location.
|
|
|
|
for (auto entry : forwardReferencePlaceholders)
|
|
|
|
errors.push_back({entry.second.getPointer(), entry.first});
|
|
|
|
llvm::array_pod_sort(errors.begin(), errors.end());
|
|
|
|
|
|
|
|
for (auto entry : errors)
|
|
|
|
emitError(SMLoc::getFromPointer(entry.first),
|
|
|
|
"use of undeclared SSA value name");
|
|
|
|
return ParseFailure;
|
|
|
|
}
|
|
|
|
|
|
|
|
return ParseSuccess;
|
2018-07-19 23:35:28 +08:00
|
|
|
}
|
|
|
|
|
2018-07-08 06:48:26 +08:00
|
|
|
/// Parse a SSA operand for an instruction or statement.
|
|
|
|
///
|
2018-07-23 06:45:24 +08:00
|
|
|
/// ssa-use ::= ssa-id
|
2018-07-08 06:48:26 +08:00
|
|
|
///
|
2018-07-19 23:35:28 +08:00
|
|
|
ParseResult FunctionParser::parseSSAUse(SSAUseInfo &result) {
|
2018-07-21 09:41:34 +08:00
|
|
|
result.name = getTokenSpelling();
|
|
|
|
result.number = 0;
|
|
|
|
result.loc = getToken().getLoc();
|
2018-07-24 08:30:01 +08:00
|
|
|
if (parseToken(Token::percent_identifier, "expected SSA operand"))
|
|
|
|
return ParseFailure;
|
2018-07-21 09:41:34 +08:00
|
|
|
|
|
|
|
// If we have an affine map ID, it is a result number.
|
|
|
|
if (getToken().is(Token::hash_identifier)) {
|
|
|
|
if (auto value = getToken().getHashIdentifierNumber())
|
|
|
|
result.number = value.getValue();
|
|
|
|
else
|
|
|
|
return emitError("invalid SSA value result number");
|
|
|
|
consumeToken(Token::hash_identifier);
|
|
|
|
}
|
|
|
|
|
2018-07-19 23:35:28 +08:00
|
|
|
return ParseSuccess;
|
2018-07-08 06:48:26 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
/// Parse a (possibly empty) list of SSA operands.
|
|
|
|
///
|
|
|
|
/// ssa-use-list ::= ssa-use (`,` ssa-use)*
|
|
|
|
/// ssa-use-list-opt ::= ssa-use-list?
|
|
|
|
///
|
2018-07-19 23:35:28 +08:00
|
|
|
ParseResult
|
2018-07-22 05:32:09 +08:00
|
|
|
FunctionParser::parseOptionalSSAUseList(SmallVectorImpl<SSAUseInfo> &results) {
|
2018-07-26 02:15:20 +08:00
|
|
|
if (getToken().isNot(Token::percent_identifier))
|
2018-07-22 05:32:09 +08:00
|
|
|
return ParseSuccess;
|
|
|
|
return parseCommaSeparatedList([&]() -> ParseResult {
|
2018-07-19 23:35:28 +08:00
|
|
|
SSAUseInfo result;
|
|
|
|
if (parseSSAUse(result))
|
|
|
|
return ParseFailure;
|
|
|
|
results.push_back(result);
|
|
|
|
return ParseSuccess;
|
|
|
|
});
|
2018-07-08 06:48:26 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
/// Parse an SSA use with an associated type.
|
|
|
|
///
|
|
|
|
/// ssa-use-and-type ::= ssa-use `:` type
|
2018-07-23 06:45:24 +08:00
|
|
|
template <typename ResultType>
|
|
|
|
ResultType FunctionParser::parseSSADefOrUseAndType(
|
|
|
|
const std::function<ResultType(SSAUseInfo, Type *)> &action) {
|
2018-07-24 08:30:01 +08:00
|
|
|
|
2018-07-19 23:35:28 +08:00
|
|
|
SSAUseInfo useInfo;
|
2018-07-24 08:30:01 +08:00
|
|
|
if (parseSSAUse(useInfo) ||
|
|
|
|
parseToken(Token::colon, "expected ':' and type for SSA operand"))
|
2018-07-19 23:35:28 +08:00
|
|
|
return nullptr;
|
2018-07-08 06:48:26 +08:00
|
|
|
|
2018-07-19 23:35:28 +08:00
|
|
|
auto *type = parseType();
|
|
|
|
if (!type)
|
|
|
|
return nullptr;
|
2018-07-08 06:48:26 +08:00
|
|
|
|
2018-07-23 06:45:24 +08:00
|
|
|
return action(useInfo, type);
|
2018-07-08 06:48:26 +08:00
|
|
|
}
|
|
|
|
|
2018-07-24 02:56:17 +08:00
|
|
|
/// Parse a (possibly empty) list of SSA operands, followed by a colon, then
|
|
|
|
/// followed by a type list. If hasParens is true, then the operands are
|
|
|
|
/// surrounded by parens.
|
2018-07-08 06:48:26 +08:00
|
|
|
///
|
2018-07-24 02:56:17 +08:00
|
|
|
/// ssa-use-and-type-list[parens]
|
|
|
|
/// ::= `(` ssa-use-list `)` ':' type-list-no-parens
|
|
|
|
///
|
|
|
|
/// ssa-use-and-type-list[!parens]
|
|
|
|
/// ::= ssa-use-list ':' type-list-no-parens
|
2018-07-08 06:48:26 +08:00
|
|
|
///
|
2018-07-22 05:32:09 +08:00
|
|
|
template <typename ValueTy>
|
2018-07-19 23:35:28 +08:00
|
|
|
ParseResult FunctionParser::parseOptionalSSAUseAndTypeList(
|
2018-07-24 02:56:17 +08:00
|
|
|
SmallVectorImpl<ValueTy *> &results, bool isParenthesized) {
|
|
|
|
|
|
|
|
// If we are in the parenthesized form and no paren exists, then we succeed
|
|
|
|
// with an empty list.
|
|
|
|
if (isParenthesized && !consumeIf(Token::l_paren))
|
2018-07-22 05:32:09 +08:00
|
|
|
return ParseSuccess;
|
|
|
|
|
2018-07-24 02:56:17 +08:00
|
|
|
SmallVector<SSAUseInfo, 4> valueIDs;
|
|
|
|
if (parseOptionalSSAUseList(valueIDs))
|
2018-07-19 23:35:28 +08:00
|
|
|
return ParseFailure;
|
2018-07-24 02:56:17 +08:00
|
|
|
|
|
|
|
if (isParenthesized && !consumeIf(Token::r_paren))
|
|
|
|
return emitError("expected ')' in operand list");
|
|
|
|
|
|
|
|
// If there were no operands, then there is no colon or type lists.
|
|
|
|
if (valueIDs.empty())
|
|
|
|
return ParseSuccess;
|
|
|
|
|
|
|
|
SmallVector<Type *, 4> types;
|
2018-07-24 08:30:01 +08:00
|
|
|
if (parseToken(Token::colon, "expected ':' in operand list") ||
|
|
|
|
parseTypeListNoParens(types))
|
2018-07-24 02:56:17 +08:00
|
|
|
return ParseFailure;
|
|
|
|
|
|
|
|
if (valueIDs.size() != types.size())
|
|
|
|
return emitError("expected " + Twine(valueIDs.size()) +
|
|
|
|
" types to match operand list");
|
|
|
|
|
|
|
|
results.reserve(valueIDs.size());
|
|
|
|
for (unsigned i = 0, e = valueIDs.size(); i != e; ++i) {
|
|
|
|
if (auto *value = resolveSSAUse(valueIDs[i], types[i]))
|
|
|
|
results.push_back(cast<ValueTy>(value));
|
|
|
|
else
|
|
|
|
return ParseFailure;
|
|
|
|
}
|
|
|
|
|
|
|
|
return ParseSuccess;
|
2018-07-08 06:48:26 +08:00
|
|
|
}
|
|
|
|
|
2018-07-17 02:47:09 +08:00
|
|
|
/// Parse the CFG or MLFunc operation.
|
|
|
|
///
|
|
|
|
/// operation ::=
|
|
|
|
/// (ssa-id `=`)? string '(' ssa-use-list? ')' attribute-dict?
|
|
|
|
/// `:` function-type
|
|
|
|
///
|
|
|
|
ParseResult
|
2018-07-19 23:35:28 +08:00
|
|
|
FunctionParser::parseOperation(const CreateOperationFunction &createOpFunc) {
|
2018-07-17 02:47:09 +08:00
|
|
|
auto loc = getToken().getLoc();
|
|
|
|
|
|
|
|
StringRef resultID;
|
|
|
|
if (getToken().is(Token::percent_identifier)) {
|
2018-07-19 23:35:28 +08:00
|
|
|
resultID = getTokenSpelling();
|
2018-07-17 02:47:09 +08:00
|
|
|
consumeToken(Token::percent_identifier);
|
2018-07-24 08:30:01 +08:00
|
|
|
if (parseToken(Token::equal, "expected '=' after SSA name"))
|
|
|
|
return ParseFailure;
|
2018-07-17 02:47:09 +08:00
|
|
|
}
|
|
|
|
|
2018-07-26 02:15:20 +08:00
|
|
|
Operation *op;
|
|
|
|
if (getToken().is(Token::bare_identifier) || getToken().isKeyword())
|
|
|
|
op = parseCustomOperation(createOpFunc);
|
|
|
|
else if (getToken().is(Token::string))
|
|
|
|
op = parseVerboseOperation(createOpFunc);
|
|
|
|
else
|
2018-07-17 02:47:09 +08:00
|
|
|
return emitError("expected operation name in quotes");
|
|
|
|
|
2018-07-26 02:15:20 +08:00
|
|
|
// If parsing of the basic operation failed, then this whole thing fails.
|
|
|
|
if (!op)
|
|
|
|
return ParseFailure;
|
|
|
|
|
|
|
|
// We just parsed an operation. If it is a recognized one, verify that it
|
|
|
|
// is structurally as we expect. If not, produce an error with a reasonable
|
|
|
|
// source location.
|
2018-08-02 01:18:59 +08:00
|
|
|
if (auto *opInfo = op->getAbstractOperation()) {
|
2018-07-26 02:15:20 +08:00
|
|
|
if (auto error = opInfo->verifyInvariants(op))
|
|
|
|
return emitError(loc, Twine("'") + op->getName().str() + "' op " + error);
|
|
|
|
}
|
|
|
|
|
|
|
|
// If the instruction had a name, register it.
|
|
|
|
if (!resultID.empty()) {
|
2018-07-27 09:09:20 +08:00
|
|
|
if (op->getNumResults() == 0)
|
|
|
|
return emitError(loc, "cannot name an operation with no results");
|
|
|
|
|
|
|
|
for (unsigned i = 0, e = op->getNumResults(); i != e; ++i)
|
2018-08-07 05:19:46 +08:00
|
|
|
if (addDefinition({resultID, i, loc}, op->getResult(i)))
|
|
|
|
return ParseFailure;
|
2018-07-26 02:15:20 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
return ParseSuccess;
|
|
|
|
}
|
|
|
|
|
|
|
|
Operation *FunctionParser::parseVerboseOperation(
|
|
|
|
const CreateOperationFunction &createOpFunc) {
|
2018-08-24 05:32:25 +08:00
|
|
|
|
|
|
|
// Get location information for the operation.
|
|
|
|
auto *srcLocation = getEncodedSourceLocation(getToken().getLoc());
|
|
|
|
|
2018-07-17 02:47:09 +08:00
|
|
|
auto name = getToken().getStringValue();
|
|
|
|
if (name.empty())
|
2018-07-26 02:15:20 +08:00
|
|
|
return (emitError("empty operation name is invalid"), nullptr);
|
2018-07-17 02:47:09 +08:00
|
|
|
|
|
|
|
consumeToken(Token::string);
|
|
|
|
|
2018-08-24 05:32:25 +08:00
|
|
|
OperationState result(builder.getContext(), srcLocation, name);
|
2018-08-08 03:02:37 +08:00
|
|
|
|
2018-07-17 02:47:09 +08:00
|
|
|
// Parse the operand list.
|
2018-07-19 23:35:28 +08:00
|
|
|
SmallVector<SSAUseInfo, 8> operandInfos;
|
2018-07-22 05:32:09 +08:00
|
|
|
|
2018-07-24 08:30:01 +08:00
|
|
|
if (parseToken(Token::l_paren, "expected '(' to start operand list") ||
|
|
|
|
parseOptionalSSAUseList(operandInfos) ||
|
|
|
|
parseToken(Token::r_paren, "expected ')' to end operand list")) {
|
2018-07-26 02:15:20 +08:00
|
|
|
return nullptr;
|
2018-07-24 08:30:01 +08:00
|
|
|
}
|
2018-07-17 02:47:09 +08:00
|
|
|
|
|
|
|
if (getToken().is(Token::l_brace)) {
|
2018-08-08 03:02:37 +08:00
|
|
|
if (parseAttributeDict(result.attributes))
|
2018-07-26 02:15:20 +08:00
|
|
|
return nullptr;
|
2018-07-17 02:47:09 +08:00
|
|
|
}
|
|
|
|
|
2018-07-24 08:30:01 +08:00
|
|
|
if (parseToken(Token::colon, "expected ':' followed by instruction type"))
|
2018-07-26 02:15:20 +08:00
|
|
|
return nullptr;
|
2018-07-19 06:31:25 +08:00
|
|
|
|
|
|
|
auto typeLoc = getToken().getLoc();
|
|
|
|
auto type = parseType();
|
|
|
|
if (!type)
|
2018-07-26 02:15:20 +08:00
|
|
|
return nullptr;
|
2018-07-19 06:31:25 +08:00
|
|
|
auto fnType = dyn_cast<FunctionType>(type);
|
|
|
|
if (!fnType)
|
2018-07-26 02:15:20 +08:00
|
|
|
return (emitError(typeLoc, "expected function type"), nullptr);
|
2018-07-19 06:31:25 +08:00
|
|
|
|
2018-08-23 10:25:49 +08:00
|
|
|
result.addTypes(fnType->getResults());
|
2018-08-08 03:02:37 +08:00
|
|
|
|
2018-07-19 23:35:28 +08:00
|
|
|
// Check that we have the right number of types for the operands.
|
|
|
|
auto operandTypes = fnType->getInputs();
|
|
|
|
if (operandTypes.size() != operandInfos.size()) {
|
|
|
|
auto plural = "s"[operandInfos.size() == 1];
|
2018-07-26 02:15:20 +08:00
|
|
|
return (emitError(typeLoc, "expected " + llvm::utostr(operandInfos.size()) +
|
|
|
|
" operand type" + plural + " but had " +
|
|
|
|
llvm::utostr(operandTypes.size())),
|
|
|
|
nullptr);
|
2018-07-19 23:35:28 +08:00
|
|
|
}
|
2018-07-17 02:47:09 +08:00
|
|
|
|
2018-07-19 23:35:28 +08:00
|
|
|
// Resolve all of the operands.
|
|
|
|
for (unsigned i = 0, e = operandInfos.size(); i != e; ++i) {
|
2018-08-08 03:02:37 +08:00
|
|
|
result.operands.push_back(resolveSSAUse(operandInfos[i], operandTypes[i]));
|
|
|
|
if (!result.operands.back())
|
2018-07-26 02:15:20 +08:00
|
|
|
return nullptr;
|
2018-07-19 23:35:28 +08:00
|
|
|
}
|
2018-07-17 02:47:09 +08:00
|
|
|
|
2018-08-08 03:02:37 +08:00
|
|
|
return createOpFunc(result);
|
2018-07-26 02:15:20 +08:00
|
|
|
}
|
2018-07-17 02:47:09 +08:00
|
|
|
|
2018-07-26 02:15:20 +08:00
|
|
|
namespace {
|
|
|
|
class CustomOpAsmParser : public OpAsmParser {
|
|
|
|
public:
|
|
|
|
CustomOpAsmParser(SMLoc nameLoc, StringRef opName, FunctionParser &parser)
|
|
|
|
: nameLoc(nameLoc), opName(opName), parser(parser) {}
|
|
|
|
|
|
|
|
//===--------------------------------------------------------------------===//
|
|
|
|
// High level parsing methods.
|
|
|
|
//===--------------------------------------------------------------------===//
|
2018-07-19 23:35:28 +08:00
|
|
|
|
2018-08-22 08:55:22 +08:00
|
|
|
bool getCurrentLocation(llvm::SMLoc *loc) override {
|
|
|
|
*loc = parser.getToken().getLoc();
|
|
|
|
return false;
|
|
|
|
}
|
2018-08-24 05:58:27 +08:00
|
|
|
bool parseComma() override {
|
2018-07-26 02:15:20 +08:00
|
|
|
return parser.parseToken(Token::comma, "expected ','");
|
|
|
|
}
|
|
|
|
|
2018-08-24 05:58:27 +08:00
|
|
|
bool parseColonType(Type *&result) override {
|
|
|
|
return parser.parseToken(Token::colon, "expected ':'") ||
|
|
|
|
!(result = parser.parseType());
|
2018-07-26 02:15:20 +08:00
|
|
|
}
|
|
|
|
|
2018-08-24 05:58:27 +08:00
|
|
|
bool parseColonTypeList(SmallVectorImpl<Type *> &result) override {
|
|
|
|
if (parser.parseToken(Token::colon, "expected ':'"))
|
2018-07-26 02:15:20 +08:00
|
|
|
return true;
|
|
|
|
|
|
|
|
do {
|
|
|
|
if (auto *type = parser.parseType())
|
|
|
|
result.push_back(type);
|
|
|
|
else
|
|
|
|
return true;
|
|
|
|
|
|
|
|
} while (parser.consumeIf(Token::comma));
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2018-08-03 07:54:36 +08:00
|
|
|
/// Parse an arbitrary attribute and return it in result. This also adds the
|
|
|
|
/// attribute to the specified attribute list with the specified name. this
|
|
|
|
/// captures the location of the attribute in 'loc' if it is non-null.
|
|
|
|
bool parseAttribute(Attribute *&result, const char *attrName,
|
2018-08-24 05:58:27 +08:00
|
|
|
SmallVectorImpl<NamedAttribute> &attrs) override {
|
2018-07-26 02:15:20 +08:00
|
|
|
result = parser.parseAttribute();
|
2018-08-03 07:54:36 +08:00
|
|
|
if (!result)
|
|
|
|
return true;
|
|
|
|
|
|
|
|
attrs.push_back(
|
|
|
|
NamedAttribute(parser.builder.getIdentifier(attrName), result));
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
/// If a named attribute list is present, parse is into result.
|
2018-08-24 05:58:27 +08:00
|
|
|
bool
|
|
|
|
parseOptionalAttributeDict(SmallVectorImpl<NamedAttribute> &result) override {
|
2018-08-03 07:54:36 +08:00
|
|
|
if (parser.getToken().isNot(Token::l_brace))
|
|
|
|
return false;
|
|
|
|
return parser.parseAttributeDict(result) == ParseFailure;
|
2018-07-26 02:15:20 +08:00
|
|
|
}
|
|
|
|
|
2018-08-22 08:55:22 +08:00
|
|
|
/// Parse a function name like '@foo' and return the name in a form that can
|
|
|
|
/// be passed to resolveFunctionName when a function type is available.
|
|
|
|
virtual bool parseFunctionName(StringRef &result, llvm::SMLoc &loc) {
|
|
|
|
loc = parser.getToken().getLoc();
|
|
|
|
|
|
|
|
if (parser.getToken().isNot(Token::at_identifier))
|
|
|
|
return emitError(loc, "expected function name");
|
|
|
|
|
|
|
|
result = parser.getTokenSpelling();
|
|
|
|
parser.consumeToken(Token::at_identifier);
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2018-07-26 02:15:20 +08:00
|
|
|
bool parseOperand(OperandType &result) override {
|
|
|
|
FunctionParser::SSAUseInfo useInfo;
|
|
|
|
if (parser.parseSSAUse(useInfo))
|
|
|
|
return true;
|
|
|
|
|
|
|
|
result = {useInfo.loc, useInfo.name, useInfo.number};
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool parseOperandList(SmallVectorImpl<OperandType> &result,
|
|
|
|
int requiredOperandCount = -1,
|
2018-08-03 07:54:36 +08:00
|
|
|
Delimiter delimiter = Delimiter::None) override {
|
2018-07-26 02:15:20 +08:00
|
|
|
auto startLoc = parser.getToken().getLoc();
|
|
|
|
|
2018-08-03 07:54:36 +08:00
|
|
|
// Handle delimiters.
|
|
|
|
switch (delimiter) {
|
|
|
|
case Delimiter::None:
|
2018-07-26 02:15:20 +08:00
|
|
|
break;
|
2018-08-03 07:54:36 +08:00
|
|
|
case Delimiter::OptionalParen:
|
2018-07-29 00:36:25 +08:00
|
|
|
if (parser.getToken().isNot(Token::l_paren))
|
|
|
|
return false;
|
|
|
|
LLVM_FALLTHROUGH;
|
2018-08-03 07:54:36 +08:00
|
|
|
case Delimiter::Paren:
|
2018-07-26 02:15:20 +08:00
|
|
|
if (parser.parseToken(Token::l_paren, "expected '(' in operand list"))
|
|
|
|
return true;
|
|
|
|
break;
|
2018-08-03 07:54:36 +08:00
|
|
|
case Delimiter::OptionalSquare:
|
2018-07-29 00:36:25 +08:00
|
|
|
if (parser.getToken().isNot(Token::l_square))
|
|
|
|
return false;
|
|
|
|
LLVM_FALLTHROUGH;
|
2018-08-03 07:54:36 +08:00
|
|
|
case Delimiter::Square:
|
2018-07-26 02:15:20 +08:00
|
|
|
if (parser.parseToken(Token::l_square, "expected '[' in operand list"))
|
|
|
|
return true;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Check for zero operands.
|
|
|
|
if (parser.getToken().is(Token::percent_identifier)) {
|
|
|
|
do {
|
|
|
|
OperandType operand;
|
|
|
|
if (parseOperand(operand))
|
|
|
|
return true;
|
|
|
|
result.push_back(operand);
|
|
|
|
} while (parser.consumeIf(Token::comma));
|
|
|
|
}
|
|
|
|
|
2018-08-03 07:54:36 +08:00
|
|
|
// Handle delimiters. If we reach here, the optional delimiters were
|
2018-07-29 00:36:25 +08:00
|
|
|
// present, so we need to parse their closing one.
|
2018-08-03 07:54:36 +08:00
|
|
|
switch (delimiter) {
|
|
|
|
case Delimiter::None:
|
2018-07-26 02:15:20 +08:00
|
|
|
break;
|
2018-08-03 07:54:36 +08:00
|
|
|
case Delimiter::OptionalParen:
|
|
|
|
case Delimiter::Paren:
|
2018-07-26 02:15:20 +08:00
|
|
|
if (parser.parseToken(Token::r_paren, "expected ')' in operand list"))
|
|
|
|
return true;
|
|
|
|
break;
|
2018-08-03 07:54:36 +08:00
|
|
|
case Delimiter::OptionalSquare:
|
|
|
|
case Delimiter::Square:
|
2018-07-26 02:15:20 +08:00
|
|
|
if (parser.parseToken(Token::r_square, "expected ']' in operand list"))
|
|
|
|
return true;
|
|
|
|
break;
|
2018-07-19 23:35:28 +08:00
|
|
|
}
|
2018-07-26 02:15:20 +08:00
|
|
|
|
|
|
|
if (requiredOperandCount != -1 && result.size() != requiredOperandCount)
|
|
|
|
emitError(startLoc,
|
|
|
|
"expected " + Twine(requiredOperandCount) + " operands");
|
|
|
|
return false;
|
2018-07-19 23:35:28 +08:00
|
|
|
}
|
|
|
|
|
2018-08-22 08:55:22 +08:00
|
|
|
/// Resolve a parse function name and a type into a function reference.
|
|
|
|
virtual bool resolveFunctionName(StringRef name, FunctionType *type,
|
|
|
|
llvm::SMLoc loc, Function *&result) {
|
|
|
|
result = parser.resolveFunctionReference(name, loc, type);
|
|
|
|
return result == nullptr;
|
|
|
|
}
|
|
|
|
|
2018-07-26 02:15:20 +08:00
|
|
|
//===--------------------------------------------------------------------===//
|
|
|
|
// Methods for interacting with the parser
|
|
|
|
//===--------------------------------------------------------------------===//
|
|
|
|
|
|
|
|
Builder &getBuilder() const override { return parser.builder; }
|
|
|
|
|
|
|
|
llvm::SMLoc getNameLoc() const override { return nameLoc; }
|
|
|
|
|
2018-08-22 08:55:22 +08:00
|
|
|
bool resolveOperand(const OperandType &operand, Type *type,
|
2018-08-09 02:02:58 +08:00
|
|
|
SmallVectorImpl<SSAValue *> &result) override {
|
2018-07-26 02:15:20 +08:00
|
|
|
FunctionParser::SSAUseInfo operandInfo = {operand.name, operand.number,
|
|
|
|
operand.location};
|
2018-08-09 02:02:58 +08:00
|
|
|
if (auto *value = parser.resolveSSAUse(operandInfo, type)) {
|
|
|
|
result.push_back(value);
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
return true;
|
2018-07-26 02:15:20 +08:00
|
|
|
}
|
|
|
|
|
2018-08-08 00:12:35 +08:00
|
|
|
/// Emit a diagnostic at the specified location and return true.
|
|
|
|
bool emitError(llvm::SMLoc loc, const Twine &message) override {
|
2018-07-26 02:15:20 +08:00
|
|
|
parser.emitError(loc, "custom op '" + Twine(opName) + "' " + message);
|
|
|
|
emittedError = true;
|
2018-08-08 00:12:35 +08:00
|
|
|
return true;
|
2018-07-26 02:15:20 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
bool didEmitError() const { return emittedError; }
|
|
|
|
|
|
|
|
private:
|
|
|
|
SMLoc nameLoc;
|
|
|
|
StringRef opName;
|
|
|
|
FunctionParser &parser;
|
|
|
|
bool emittedError = false;
|
|
|
|
};
|
|
|
|
} // end anonymous namespace.
|
|
|
|
|
|
|
|
Operation *FunctionParser::parseCustomOperation(
|
|
|
|
const CreateOperationFunction &createOpFunc) {
|
|
|
|
auto opLoc = getToken().getLoc();
|
|
|
|
auto opName = getTokenSpelling();
|
|
|
|
CustomOpAsmParser opAsmParser(opLoc, opName, *this);
|
|
|
|
|
|
|
|
auto *opDefinition = getOperationSet().lookup(opName);
|
|
|
|
if (!opDefinition) {
|
|
|
|
opAsmParser.emitError(opLoc, "is unknown");
|
|
|
|
return nullptr;
|
|
|
|
}
|
|
|
|
|
|
|
|
consumeToken();
|
|
|
|
|
2018-08-22 08:55:22 +08:00
|
|
|
// If the custom op parser crashes, produce some indication to help debugging.
|
|
|
|
std::string opNameStr = opName.str();
|
|
|
|
llvm::PrettyStackTraceFormat fmt("MLIR Parser: custom op parser '%s'",
|
|
|
|
opNameStr.c_str());
|
|
|
|
|
2018-08-24 05:32:25 +08:00
|
|
|
// Get location information for the operation.
|
|
|
|
auto *srcLocation = getEncodedSourceLocation(opLoc);
|
|
|
|
|
2018-07-26 02:15:20 +08:00
|
|
|
// Have the op implementation take a crack and parsing this.
|
2018-08-24 05:32:25 +08:00
|
|
|
OperationState opState(builder.getContext(), srcLocation, opName);
|
2018-08-08 00:12:35 +08:00
|
|
|
if (opDefinition->parseAssembly(&opAsmParser, &opState))
|
|
|
|
return nullptr;
|
2018-07-26 02:15:20 +08:00
|
|
|
|
|
|
|
// If it emitted an error, we failed.
|
|
|
|
if (opAsmParser.didEmitError())
|
|
|
|
return nullptr;
|
|
|
|
|
|
|
|
// Otherwise, we succeeded. Use the state it parsed as our op information.
|
2018-08-08 03:02:37 +08:00
|
|
|
return createOpFunc(opState);
|
2018-07-17 02:47:09 +08:00
|
|
|
}
|
2018-06-24 07:03:42 +08:00
|
|
|
|
2018-07-10 10:05:38 +08:00
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
// CFG Functions
|
|
|
|
//===----------------------------------------------------------------------===//
|
2018-06-24 07:03:42 +08:00
|
|
|
|
|
|
|
namespace {
|
2018-07-10 10:05:38 +08:00
|
|
|
/// This is a specialized parser for CFGFunction's, maintaining the state
|
|
|
|
/// transient to their bodies.
|
2018-07-19 23:35:28 +08:00
|
|
|
class CFGFunctionParser : public FunctionParser {
|
2018-07-09 11:51:38 +08:00
|
|
|
public:
|
2018-07-11 01:08:27 +08:00
|
|
|
CFGFunctionParser(ParserState &state, CFGFunction *function)
|
2018-07-27 09:09:20 +08:00
|
|
|
: FunctionParser(state, Kind::CFGFunc), function(function),
|
|
|
|
builder(function) {}
|
2018-07-11 01:08:27 +08:00
|
|
|
|
|
|
|
ParseResult parseFunctionBody();
|
|
|
|
|
|
|
|
private:
|
2018-06-25 02:18:29 +08:00
|
|
|
CFGFunction *function;
|
2018-07-24 07:56:32 +08:00
|
|
|
llvm::StringMap<std::pair<BasicBlock *, SMLoc>> blocksByName;
|
2018-07-10 10:05:38 +08:00
|
|
|
|
|
|
|
/// This builder intentionally shadows the builder in the base class, with a
|
|
|
|
/// more specific builder type.
|
2018-07-09 11:51:38 +08:00
|
|
|
CFGFuncBuilder builder;
|
2018-06-25 02:18:29 +08:00
|
|
|
|
2018-06-24 07:03:42 +08:00
|
|
|
/// Get the basic block with the specified name, creating it if it doesn't
|
2018-06-25 02:18:29 +08:00
|
|
|
/// already exist. The location specified is the point of use, which allows
|
|
|
|
/// us to diagnose references to blocks that are not defined precisely.
|
|
|
|
BasicBlock *getBlockNamed(StringRef name, SMLoc loc) {
|
|
|
|
auto &blockAndLoc = blocksByName[name];
|
|
|
|
if (!blockAndLoc.first) {
|
2018-07-02 11:28:00 +08:00
|
|
|
blockAndLoc.first = new BasicBlock();
|
2018-06-25 02:18:29 +08:00
|
|
|
blockAndLoc.second = loc;
|
2018-06-24 07:03:42 +08:00
|
|
|
}
|
2018-06-25 02:18:29 +08:00
|
|
|
return blockAndLoc.first;
|
2018-06-24 07:03:42 +08:00
|
|
|
}
|
2018-07-10 10:05:38 +08:00
|
|
|
|
2018-07-23 06:45:24 +08:00
|
|
|
ParseResult
|
|
|
|
parseOptionalBasicBlockArgList(SmallVectorImpl<BBArgument *> &results,
|
|
|
|
BasicBlock *owner);
|
2018-07-25 06:01:27 +08:00
|
|
|
ParseResult parseBranchBlockAndUseList(BasicBlock *&block,
|
|
|
|
SmallVectorImpl<CFGValue *> &values);
|
2018-07-23 06:45:24 +08:00
|
|
|
|
2018-07-10 10:05:38 +08:00
|
|
|
ParseResult parseBasicBlock();
|
|
|
|
OperationInst *parseCFGOperation();
|
|
|
|
TerminatorInst *parseTerminator();
|
2018-06-24 07:03:42 +08:00
|
|
|
};
|
|
|
|
} // end anonymous namespace
|
|
|
|
|
2018-07-23 06:45:24 +08:00
|
|
|
/// Parse a (possibly empty) list of SSA operands with types as basic block
|
2018-07-24 02:56:17 +08:00
|
|
|
/// arguments.
|
2018-07-23 06:45:24 +08:00
|
|
|
///
|
|
|
|
/// ssa-id-and-type-list ::= ssa-id-and-type (`,` ssa-id-and-type)*
|
|
|
|
///
|
|
|
|
ParseResult CFGFunctionParser::parseOptionalBasicBlockArgList(
|
|
|
|
SmallVectorImpl<BBArgument *> &results, BasicBlock *owner) {
|
|
|
|
if (getToken().is(Token::r_brace))
|
|
|
|
return ParseSuccess;
|
|
|
|
|
|
|
|
return parseCommaSeparatedList([&]() -> ParseResult {
|
|
|
|
auto type = parseSSADefOrUseAndType<Type *>(
|
|
|
|
[&](SSAUseInfo useInfo, Type *type) -> Type * {
|
|
|
|
BBArgument *arg = owner->addArgument(type);
|
2018-08-07 05:19:46 +08:00
|
|
|
if (addDefinition(useInfo, arg))
|
2018-07-23 06:45:24 +08:00
|
|
|
return nullptr;
|
|
|
|
return type;
|
|
|
|
});
|
|
|
|
return type ? ParseSuccess : ParseFailure;
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
2018-07-10 10:05:38 +08:00
|
|
|
ParseResult CFGFunctionParser::parseFunctionBody() {
|
2018-07-22 05:32:09 +08:00
|
|
|
auto braceLoc = getToken().getLoc();
|
2018-07-24 08:30:01 +08:00
|
|
|
if (parseToken(Token::l_brace, "expected '{' in CFG function"))
|
|
|
|
return ParseFailure;
|
2018-07-10 10:05:38 +08:00
|
|
|
|
2018-06-24 07:03:42 +08:00
|
|
|
// Make sure we have at least one block.
|
2018-07-10 10:05:38 +08:00
|
|
|
if (getToken().is(Token::r_brace))
|
2018-06-24 07:03:42 +08:00
|
|
|
return emitError("CFG functions must have at least one basic block");
|
|
|
|
|
|
|
|
// Parse the list of blocks.
|
|
|
|
while (!consumeIf(Token::r_brace))
|
2018-07-10 10:05:38 +08:00
|
|
|
if (parseBasicBlock())
|
2018-06-24 07:03:42 +08:00
|
|
|
return ParseFailure;
|
|
|
|
|
2018-06-25 02:18:29 +08:00
|
|
|
// Verify that all referenced blocks were defined. Iteration over a
|
|
|
|
// StringMap isn't determinstic, but this is good enough for our purposes.
|
2018-07-10 10:05:38 +08:00
|
|
|
for (auto &elt : blocksByName) {
|
2018-06-25 02:18:29 +08:00
|
|
|
auto *bb = elt.second.first;
|
2018-07-02 11:28:00 +08:00
|
|
|
if (!bb->getFunction())
|
2018-06-25 02:18:29 +08:00
|
|
|
return emitError(elt.second.second,
|
2018-07-24 07:56:32 +08:00
|
|
|
"reference to an undefined basic block '" + elt.first() +
|
|
|
|
"'");
|
2018-06-25 02:18:29 +08:00
|
|
|
}
|
|
|
|
|
2018-07-22 05:32:09 +08:00
|
|
|
return finalizeFunction(function, braceLoc);
|
2018-06-24 07:03:42 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
/// Basic block declaration.
|
|
|
|
///
|
|
|
|
/// basic-block ::= bb-label instruction* terminator-stmt
|
|
|
|
/// bb-label ::= bb-id bb-arg-list? `:`
|
|
|
|
/// bb-id ::= bare-id
|
|
|
|
/// bb-arg-list ::= `(` ssa-id-and-type-list? `)`
|
|
|
|
///
|
2018-07-10 10:05:38 +08:00
|
|
|
ParseResult CFGFunctionParser::parseBasicBlock() {
|
|
|
|
SMLoc nameLoc = getToken().getLoc();
|
|
|
|
auto name = getTokenSpelling();
|
2018-07-24 08:30:01 +08:00
|
|
|
if (parseToken(Token::bare_identifier, "expected basic block name"))
|
|
|
|
return ParseFailure;
|
2018-06-25 02:18:29 +08:00
|
|
|
|
2018-07-10 10:05:38 +08:00
|
|
|
auto *block = getBlockNamed(name, nameLoc);
|
2018-06-24 07:03:42 +08:00
|
|
|
|
|
|
|
// If this block has already been parsed, then this is a redefinition with the
|
|
|
|
// same block name.
|
2018-07-02 11:28:00 +08:00
|
|
|
if (block->getFunction())
|
2018-06-25 02:18:29 +08:00
|
|
|
return emitError(nameLoc, "redefinition of block '" + name.str() + "'");
|
|
|
|
|
2018-07-08 06:48:26 +08:00
|
|
|
// If an argument list is present, parse it.
|
|
|
|
if (consumeIf(Token::l_paren)) {
|
2018-07-23 06:45:24 +08:00
|
|
|
SmallVector<BBArgument *, 8> bbArgs;
|
2018-07-24 08:30:01 +08:00
|
|
|
if (parseOptionalBasicBlockArgList(bbArgs, block) ||
|
|
|
|
parseToken(Token::r_paren, "expected ')' to end argument list"))
|
2018-07-08 06:48:26 +08:00
|
|
|
return ParseFailure;
|
|
|
|
}
|
2018-06-24 07:03:42 +08:00
|
|
|
|
2018-07-23 06:45:24 +08:00
|
|
|
// Add the block to the function.
|
|
|
|
function->push_back(block);
|
|
|
|
|
2018-07-24 08:30:01 +08:00
|
|
|
if (parseToken(Token::colon, "expected ':' after basic block name"))
|
|
|
|
return ParseFailure;
|
2018-06-24 07:03:42 +08:00
|
|
|
|
2018-07-09 11:51:38 +08:00
|
|
|
// Set the insertion point to the block we want to insert new operations into.
|
2018-07-10 10:05:38 +08:00
|
|
|
builder.setInsertionPoint(block);
|
2018-07-09 11:51:38 +08:00
|
|
|
|
2018-08-08 03:02:37 +08:00
|
|
|
auto createOpFunc = [&](const OperationState &result) -> Operation * {
|
|
|
|
return builder.createOperation(result);
|
2018-07-17 02:47:09 +08:00
|
|
|
};
|
|
|
|
|
2018-06-29 11:45:33 +08:00
|
|
|
// Parse the list of operations that make up the body of the block.
|
2018-07-25 06:01:27 +08:00
|
|
|
while (getToken().isNot(Token::kw_return, Token::kw_br, Token::kw_cond_br)) {
|
2018-07-17 02:47:09 +08:00
|
|
|
if (parseOperation(createOpFunc))
|
2018-06-29 11:45:33 +08:00
|
|
|
return ParseFailure;
|
|
|
|
}
|
2018-06-24 07:03:42 +08:00
|
|
|
|
2018-07-17 02:47:09 +08:00
|
|
|
if (!parseTerminator())
|
2018-06-29 11:45:33 +08:00
|
|
|
return ParseFailure;
|
2018-06-24 07:03:42 +08:00
|
|
|
|
2018-06-29 11:45:33 +08:00
|
|
|
return ParseSuccess;
|
|
|
|
}
|
2018-06-24 07:03:42 +08:00
|
|
|
|
2018-07-25 06:01:27 +08:00
|
|
|
ParseResult CFGFunctionParser::parseBranchBlockAndUseList(
|
|
|
|
BasicBlock *&block, SmallVectorImpl<CFGValue *> &values) {
|
|
|
|
block = getBlockNamed(getTokenSpelling(), getToken().getLoc());
|
|
|
|
if (parseToken(Token::bare_identifier, "expected basic block name"))
|
|
|
|
return ParseFailure;
|
|
|
|
|
|
|
|
if (!consumeIf(Token::l_paren))
|
|
|
|
return ParseSuccess;
|
|
|
|
if (parseOptionalSSAUseAndTypeList(values, /*isParenthesized*/ false) ||
|
|
|
|
parseToken(Token::r_paren, "expected ')' to close argument list"))
|
|
|
|
return ParseFailure;
|
|
|
|
return ParseSuccess;
|
|
|
|
}
|
|
|
|
|
2018-06-25 02:18:29 +08:00
|
|
|
/// Parse the terminator instruction for a basic block.
|
|
|
|
///
|
|
|
|
/// terminator-stmt ::= `br` bb-id branch-use-list?
|
2018-07-23 23:42:19 +08:00
|
|
|
/// branch-use-list ::= `(` ssa-use-list `)` ':' type-list-no-parens
|
2018-06-25 02:18:29 +08:00
|
|
|
/// terminator-stmt ::=
|
|
|
|
/// `cond_br` ssa-use `,` bb-id branch-use-list? `,` bb-id branch-use-list?
|
|
|
|
/// terminator-stmt ::= `return` ssa-use-and-type-list?
|
|
|
|
///
|
2018-07-10 10:05:38 +08:00
|
|
|
TerminatorInst *CFGFunctionParser::parseTerminator() {
|
2018-08-24 05:32:25 +08:00
|
|
|
auto loc = getToken().getLoc();
|
|
|
|
|
2018-07-10 10:05:38 +08:00
|
|
|
switch (getToken().getKind()) {
|
2018-06-25 02:18:29 +08:00
|
|
|
default:
|
2018-07-02 11:28:00 +08:00
|
|
|
return (emitError("expected terminator at end of basic block"), nullptr);
|
2018-06-25 02:18:29 +08:00
|
|
|
|
2018-07-22 05:32:09 +08:00
|
|
|
case Token::kw_return: {
|
2018-06-25 02:18:29 +08:00
|
|
|
consumeToken(Token::kw_return);
|
2018-07-22 05:32:09 +08:00
|
|
|
|
2018-07-24 02:56:17 +08:00
|
|
|
// Parse any operands.
|
|
|
|
SmallVector<CFGValue *, 8> operands;
|
|
|
|
if (parseOptionalSSAUseAndTypeList(operands, /*isParenthesized*/ false))
|
|
|
|
return nullptr;
|
2018-08-24 05:58:27 +08:00
|
|
|
return builder.createReturn(getEncodedSourceLocation(loc), operands);
|
2018-07-22 05:32:09 +08:00
|
|
|
}
|
2018-06-25 02:18:29 +08:00
|
|
|
|
|
|
|
case Token::kw_br: {
|
|
|
|
consumeToken(Token::kw_br);
|
2018-07-25 06:01:27 +08:00
|
|
|
BasicBlock *destBB;
|
|
|
|
SmallVector<CFGValue *, 4> values;
|
|
|
|
if (parseBranchBlockAndUseList(destBB, values))
|
2018-07-24 08:30:01 +08:00
|
|
|
return nullptr;
|
2018-08-24 05:58:27 +08:00
|
|
|
auto branch = builder.createBranch(getEncodedSourceLocation(loc), destBB);
|
2018-07-25 06:01:27 +08:00
|
|
|
branch->addOperands(values);
|
|
|
|
return branch;
|
|
|
|
}
|
2018-07-23 23:42:19 +08:00
|
|
|
|
2018-07-25 06:01:27 +08:00
|
|
|
case Token::kw_cond_br: {
|
|
|
|
consumeToken(Token::kw_cond_br);
|
|
|
|
SSAUseInfo ssaUse;
|
|
|
|
if (parseSSAUse(ssaUse))
|
|
|
|
return nullptr;
|
|
|
|
auto *cond = resolveSSAUse(ssaUse, builder.getIntegerType(1));
|
|
|
|
if (!cond)
|
|
|
|
return (emitError("expected type was boolean (i1)"), nullptr);
|
|
|
|
if (parseToken(Token::comma, "expected ',' in conditional branch"))
|
|
|
|
return nullptr;
|
|
|
|
|
|
|
|
BasicBlock *trueBlock;
|
|
|
|
SmallVector<CFGValue *, 4> trueOperands;
|
|
|
|
if (parseBranchBlockAndUseList(trueBlock, trueOperands))
|
2018-07-23 23:42:19 +08:00
|
|
|
return nullptr;
|
2018-07-25 06:01:27 +08:00
|
|
|
|
|
|
|
if (parseToken(Token::comma, "expected ',' in conditional branch"))
|
|
|
|
return nullptr;
|
|
|
|
|
|
|
|
BasicBlock *falseBlock;
|
|
|
|
SmallVector<CFGValue *, 4> falseOperands;
|
|
|
|
if (parseBranchBlockAndUseList(falseBlock, falseOperands))
|
|
|
|
return nullptr;
|
|
|
|
|
2018-08-24 05:58:27 +08:00
|
|
|
auto branch =
|
|
|
|
builder.createCondBranch(getEncodedSourceLocation(loc),
|
|
|
|
cast<CFGValue>(cond), trueBlock, falseBlock);
|
2018-07-25 06:01:27 +08:00
|
|
|
branch->addTrueOperands(trueOperands);
|
|
|
|
branch->addFalseOperands(falseOperands);
|
2018-07-23 23:42:19 +08:00
|
|
|
return branch;
|
2018-06-25 02:18:29 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-07-10 10:05:38 +08:00
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
// ML Functions
|
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
|
|
|
|
namespace {
|
|
|
|
/// Refined parser for MLFunction bodies.
|
2018-07-19 23:35:28 +08:00
|
|
|
class MLFunctionParser : public FunctionParser {
|
2018-07-10 10:05:38 +08:00
|
|
|
public:
|
|
|
|
MLFunctionParser(ParserState &state, MLFunction *function)
|
2018-07-27 09:09:20 +08:00
|
|
|
: FunctionParser(state, Kind::MLFunc), function(function),
|
2018-08-09 02:14:57 +08:00
|
|
|
builder(function, function->end()) {}
|
2018-07-10 10:05:38 +08:00
|
|
|
|
|
|
|
ParseResult parseFunctionBody();
|
2018-07-14 04:03:13 +08:00
|
|
|
|
|
|
|
private:
|
2018-07-17 02:47:09 +08:00
|
|
|
MLFunction *function;
|
|
|
|
|
|
|
|
/// This builder intentionally shadows the builder in the base class, with a
|
|
|
|
/// more specific builder type.
|
|
|
|
MLFuncBuilder builder;
|
|
|
|
|
|
|
|
ParseResult parseForStmt();
|
2018-07-20 00:52:39 +08:00
|
|
|
AffineConstantExpr *parseIntConstant();
|
2018-07-17 02:47:09 +08:00
|
|
|
ParseResult parseIfStmt();
|
2018-07-14 04:03:13 +08:00
|
|
|
ParseResult parseElseClause(IfClause *elseClause);
|
2018-08-08 05:24:38 +08:00
|
|
|
IntegerSet *parseCondition();
|
2018-07-17 02:47:09 +08:00
|
|
|
ParseResult parseStatements(StmtBlock *block);
|
2018-07-14 04:03:13 +08:00
|
|
|
ParseResult parseStmtBlock(StmtBlock *block);
|
2018-07-10 10:05:38 +08:00
|
|
|
};
|
|
|
|
} // end anonymous namespace
|
|
|
|
|
|
|
|
ParseResult MLFunctionParser::parseFunctionBody() {
|
2018-07-22 05:32:09 +08:00
|
|
|
auto braceLoc = getToken().getLoc();
|
2018-06-29 08:02:32 +08:00
|
|
|
|
2018-08-10 03:28:58 +08:00
|
|
|
// Parse statements in this function.
|
|
|
|
if (parseStmtBlock(function))
|
2018-07-24 08:30:01 +08:00
|
|
|
return ParseFailure;
|
2018-07-22 05:32:09 +08:00
|
|
|
|
|
|
|
return finalizeFunction(function, braceLoc);
|
2018-06-29 08:02:32 +08:00
|
|
|
}
|
|
|
|
|
2018-07-04 08:51:28 +08:00
|
|
|
/// For statement.
|
|
|
|
///
|
2018-07-10 10:05:38 +08:00
|
|
|
/// ml-for-stmt ::= `for` ssa-id `=` lower-bound `to` upper-bound
|
|
|
|
/// (`step` integer-literal)? `{` ml-stmt* `}`
|
2018-07-04 08:51:28 +08:00
|
|
|
///
|
2018-07-17 02:47:09 +08:00
|
|
|
ParseResult MLFunctionParser::parseForStmt() {
|
2018-07-04 08:51:28 +08:00
|
|
|
consumeToken(Token::kw_for);
|
|
|
|
|
Extend loop unrolling to unroll by a given factor; add builder for affine
apply op.
- add builder for AffineApplyOp (first one for an operation that has
non-zero operands)
- add support for loop unrolling by a given factor; uses the affine apply op
builder.
While on this, change 'step' of ForStmt to be 'unsigned' instead of
AffineConstantExpr *. Add setters for ForStmt lb, ub, step.
Sample Input:
// CHECK-LABEL: mlfunc @loop_nest_unroll_cleanup() {
mlfunc @loop_nest_unroll_cleanup() {
for %i = 1 to 100 {
for %j = 0 to 17 {
%x = "addi32"(%j, %j) : (affineint, affineint) -> i32
%y = "addi32"(%x, %x) : (i32, i32) -> i32
}
}
return
}
Output:
$ mlir-opt -loop-unroll -unroll-factor=4 /tmp/single2.mlir
#map0 = (d0) -> (d0 + 1)
#map1 = (d0) -> (d0 + 2)
#map2 = (d0) -> (d0 + 3)
mlfunc @loop_nest_unroll_cleanup() {
for %i0 = 1 to 100 {
for %i1 = 0 to 17 step 4 {
%0 = "addi32"(%i1, %i1) : (affineint, affineint) -> i32
%1 = "addi32"(%0, %0) : (i32, i32) -> i32
%2 = affine_apply #map0(%i1)
%3 = "addi32"(%2, %2) : (affineint, affineint) -> i32
%4 = affine_apply #map1(%i1)
%5 = "addi32"(%4, %4) : (affineint, affineint) -> i32
%6 = affine_apply #map2(%i1)
%7 = "addi32"(%6, %6) : (affineint, affineint) -> i32
}
for %i2 = 16 to 17 {
%8 = "addi32"(%i2, %i2) : (affineint, affineint) -> i32
%9 = "addi32"(%8, %8) : (i32, i32) -> i32
}
}
return
}
PiperOrigin-RevId: 209676220
2018-08-22 07:01:23 +08:00
|
|
|
// Parse induction variable.
|
2018-07-20 00:52:39 +08:00
|
|
|
if (getToken().isNot(Token::percent_identifier))
|
|
|
|
return emitError("expected SSA identifier for the loop variable");
|
|
|
|
|
2018-07-31 06:18:10 +08:00
|
|
|
auto loc = getToken().getLoc();
|
2018-08-01 14:14:16 +08:00
|
|
|
StringRef inductionVariableName = getTokenSpelling();
|
2018-07-20 00:52:39 +08:00
|
|
|
consumeToken(Token::percent_identifier);
|
|
|
|
|
2018-08-10 03:28:58 +08:00
|
|
|
if (parseToken(Token::equal, "expected '='"))
|
2018-07-24 08:30:01 +08:00
|
|
|
return ParseFailure;
|
2018-07-20 00:52:39 +08:00
|
|
|
|
Extend loop unrolling to unroll by a given factor; add builder for affine
apply op.
- add builder for AffineApplyOp (first one for an operation that has
non-zero operands)
- add support for loop unrolling by a given factor; uses the affine apply op
builder.
While on this, change 'step' of ForStmt to be 'unsigned' instead of
AffineConstantExpr *. Add setters for ForStmt lb, ub, step.
Sample Input:
// CHECK-LABEL: mlfunc @loop_nest_unroll_cleanup() {
mlfunc @loop_nest_unroll_cleanup() {
for %i = 1 to 100 {
for %j = 0 to 17 {
%x = "addi32"(%j, %j) : (affineint, affineint) -> i32
%y = "addi32"(%x, %x) : (i32, i32) -> i32
}
}
return
}
Output:
$ mlir-opt -loop-unroll -unroll-factor=4 /tmp/single2.mlir
#map0 = (d0) -> (d0 + 1)
#map1 = (d0) -> (d0 + 2)
#map2 = (d0) -> (d0 + 3)
mlfunc @loop_nest_unroll_cleanup() {
for %i0 = 1 to 100 {
for %i1 = 0 to 17 step 4 {
%0 = "addi32"(%i1, %i1) : (affineint, affineint) -> i32
%1 = "addi32"(%0, %0) : (i32, i32) -> i32
%2 = affine_apply #map0(%i1)
%3 = "addi32"(%2, %2) : (affineint, affineint) -> i32
%4 = affine_apply #map1(%i1)
%5 = "addi32"(%4, %4) : (affineint, affineint) -> i32
%6 = affine_apply #map2(%i1)
%7 = "addi32"(%6, %6) : (affineint, affineint) -> i32
}
for %i2 = 16 to 17 {
%8 = "addi32"(%i2, %i2) : (affineint, affineint) -> i32
%9 = "addi32"(%8, %8) : (i32, i32) -> i32
}
}
return
}
PiperOrigin-RevId: 209676220
2018-08-22 07:01:23 +08:00
|
|
|
// Parse loop bounds.
|
2018-07-20 00:52:39 +08:00
|
|
|
AffineConstantExpr *lowerBound = parseIntConstant();
|
|
|
|
if (!lowerBound)
|
|
|
|
return ParseFailure;
|
|
|
|
|
2018-07-24 08:30:01 +08:00
|
|
|
if (parseToken(Token::kw_to, "expected 'to' between bounds"))
|
|
|
|
return ParseFailure;
|
2018-07-20 00:52:39 +08:00
|
|
|
|
|
|
|
AffineConstantExpr *upperBound = parseIntConstant();
|
|
|
|
if (!upperBound)
|
|
|
|
return ParseFailure;
|
|
|
|
|
Extend loop unrolling to unroll by a given factor; add builder for affine
apply op.
- add builder for AffineApplyOp (first one for an operation that has
non-zero operands)
- add support for loop unrolling by a given factor; uses the affine apply op
builder.
While on this, change 'step' of ForStmt to be 'unsigned' instead of
AffineConstantExpr *. Add setters for ForStmt lb, ub, step.
Sample Input:
// CHECK-LABEL: mlfunc @loop_nest_unroll_cleanup() {
mlfunc @loop_nest_unroll_cleanup() {
for %i = 1 to 100 {
for %j = 0 to 17 {
%x = "addi32"(%j, %j) : (affineint, affineint) -> i32
%y = "addi32"(%x, %x) : (i32, i32) -> i32
}
}
return
}
Output:
$ mlir-opt -loop-unroll -unroll-factor=4 /tmp/single2.mlir
#map0 = (d0) -> (d0 + 1)
#map1 = (d0) -> (d0 + 2)
#map2 = (d0) -> (d0 + 3)
mlfunc @loop_nest_unroll_cleanup() {
for %i0 = 1 to 100 {
for %i1 = 0 to 17 step 4 {
%0 = "addi32"(%i1, %i1) : (affineint, affineint) -> i32
%1 = "addi32"(%0, %0) : (i32, i32) -> i32
%2 = affine_apply #map0(%i1)
%3 = "addi32"(%2, %2) : (affineint, affineint) -> i32
%4 = affine_apply #map1(%i1)
%5 = "addi32"(%4, %4) : (affineint, affineint) -> i32
%6 = affine_apply #map2(%i1)
%7 = "addi32"(%6, %6) : (affineint, affineint) -> i32
}
for %i2 = 16 to 17 {
%8 = "addi32"(%i2, %i2) : (affineint, affineint) -> i32
%9 = "addi32"(%8, %8) : (i32, i32) -> i32
}
}
return
}
PiperOrigin-RevId: 209676220
2018-08-22 07:01:23 +08:00
|
|
|
// Parse step.
|
|
|
|
int64_t step = 1;
|
2018-07-20 00:52:39 +08:00
|
|
|
if (consumeIf(Token::kw_step)) {
|
Extend loop unrolling to unroll by a given factor; add builder for affine
apply op.
- add builder for AffineApplyOp (first one for an operation that has
non-zero operands)
- add support for loop unrolling by a given factor; uses the affine apply op
builder.
While on this, change 'step' of ForStmt to be 'unsigned' instead of
AffineConstantExpr *. Add setters for ForStmt lb, ub, step.
Sample Input:
// CHECK-LABEL: mlfunc @loop_nest_unroll_cleanup() {
mlfunc @loop_nest_unroll_cleanup() {
for %i = 1 to 100 {
for %j = 0 to 17 {
%x = "addi32"(%j, %j) : (affineint, affineint) -> i32
%y = "addi32"(%x, %x) : (i32, i32) -> i32
}
}
return
}
Output:
$ mlir-opt -loop-unroll -unroll-factor=4 /tmp/single2.mlir
#map0 = (d0) -> (d0 + 1)
#map1 = (d0) -> (d0 + 2)
#map2 = (d0) -> (d0 + 3)
mlfunc @loop_nest_unroll_cleanup() {
for %i0 = 1 to 100 {
for %i1 = 0 to 17 step 4 {
%0 = "addi32"(%i1, %i1) : (affineint, affineint) -> i32
%1 = "addi32"(%0, %0) : (i32, i32) -> i32
%2 = affine_apply #map0(%i1)
%3 = "addi32"(%2, %2) : (affineint, affineint) -> i32
%4 = affine_apply #map1(%i1)
%5 = "addi32"(%4, %4) : (affineint, affineint) -> i32
%6 = affine_apply #map2(%i1)
%7 = "addi32"(%6, %6) : (affineint, affineint) -> i32
}
for %i2 = 16 to 17 {
%8 = "addi32"(%i2, %i2) : (affineint, affineint) -> i32
%9 = "addi32"(%8, %8) : (i32, i32) -> i32
}
}
return
}
PiperOrigin-RevId: 209676220
2018-08-22 07:01:23 +08:00
|
|
|
AffineConstantExpr *stepExpr = parseIntConstant();
|
|
|
|
if (!stepExpr)
|
2018-07-20 00:52:39 +08:00
|
|
|
return ParseFailure;
|
Extend loop unrolling to unroll by a given factor; add builder for affine
apply op.
- add builder for AffineApplyOp (first one for an operation that has
non-zero operands)
- add support for loop unrolling by a given factor; uses the affine apply op
builder.
While on this, change 'step' of ForStmt to be 'unsigned' instead of
AffineConstantExpr *. Add setters for ForStmt lb, ub, step.
Sample Input:
// CHECK-LABEL: mlfunc @loop_nest_unroll_cleanup() {
mlfunc @loop_nest_unroll_cleanup() {
for %i = 1 to 100 {
for %j = 0 to 17 {
%x = "addi32"(%j, %j) : (affineint, affineint) -> i32
%y = "addi32"(%x, %x) : (i32, i32) -> i32
}
}
return
}
Output:
$ mlir-opt -loop-unroll -unroll-factor=4 /tmp/single2.mlir
#map0 = (d0) -> (d0 + 1)
#map1 = (d0) -> (d0 + 2)
#map2 = (d0) -> (d0 + 3)
mlfunc @loop_nest_unroll_cleanup() {
for %i0 = 1 to 100 {
for %i1 = 0 to 17 step 4 {
%0 = "addi32"(%i1, %i1) : (affineint, affineint) -> i32
%1 = "addi32"(%0, %0) : (i32, i32) -> i32
%2 = affine_apply #map0(%i1)
%3 = "addi32"(%2, %2) : (affineint, affineint) -> i32
%4 = affine_apply #map1(%i1)
%5 = "addi32"(%4, %4) : (affineint, affineint) -> i32
%6 = affine_apply #map2(%i1)
%7 = "addi32"(%6, %6) : (affineint, affineint) -> i32
}
for %i2 = 16 to 17 {
%8 = "addi32"(%i2, %i2) : (affineint, affineint) -> i32
%9 = "addi32"(%8, %8) : (i32, i32) -> i32
}
}
return
}
PiperOrigin-RevId: 209676220
2018-08-22 07:01:23 +08:00
|
|
|
step = stepExpr->getValue();
|
2018-07-20 00:52:39 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
// Create for statement.
|
2018-08-24 05:32:25 +08:00
|
|
|
ForStmt *forStmt = builder.createFor(getEncodedSourceLocation(loc),
|
|
|
|
lowerBound, upperBound, step);
|
2018-07-31 06:18:10 +08:00
|
|
|
|
|
|
|
// Create SSA value definition for the induction variable.
|
2018-08-07 05:19:46 +08:00
|
|
|
if (addDefinition({inductionVariableName, 0, loc}, forStmt))
|
|
|
|
return ParseFailure;
|
2018-07-17 02:47:09 +08:00
|
|
|
|
2018-07-20 00:52:39 +08:00
|
|
|
// If parsing of the for statement body fails,
|
|
|
|
// MLIR contains for statement with those nested statements that have been
|
|
|
|
// successfully parsed.
|
2018-07-31 06:18:10 +08:00
|
|
|
if (parseStmtBlock(forStmt))
|
2018-07-17 02:47:09 +08:00
|
|
|
return ParseFailure;
|
|
|
|
|
2018-07-31 06:18:10 +08:00
|
|
|
// Reset insertion point to the current block.
|
2018-08-09 02:14:57 +08:00
|
|
|
builder.setInsertionPointToEnd(forStmt->getBlock());
|
2018-07-17 02:47:09 +08:00
|
|
|
return ParseSuccess;
|
2018-07-04 08:51:28 +08:00
|
|
|
}
|
|
|
|
|
2018-07-20 00:52:39 +08:00
|
|
|
// This method is temporary workaround to parse simple loop bounds and
|
|
|
|
// step.
|
|
|
|
// TODO: remove this method once it's no longer used.
|
|
|
|
AffineConstantExpr *MLFunctionParser::parseIntConstant() {
|
|
|
|
if (getToken().isNot(Token::integer))
|
|
|
|
return (emitError("expected non-negative integer for now"), nullptr);
|
|
|
|
|
|
|
|
auto val = getToken().getUInt64IntegerValue();
|
|
|
|
if (!val.hasValue() || (int64_t)val.getValue() < 0) {
|
|
|
|
return (emitError("constant too large for affineint"), nullptr);
|
|
|
|
}
|
|
|
|
consumeToken(Token::integer);
|
|
|
|
return builder.getConstantExpr((int64_t)val.getValue());
|
|
|
|
}
|
|
|
|
|
2018-08-08 05:24:38 +08:00
|
|
|
/// Parse condition.
|
|
|
|
IntegerSet *MLFunctionParser::parseCondition() {
|
|
|
|
return parseIntegerSetReference();
|
|
|
|
|
|
|
|
// TODO: Parse operands to the integer set.
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Parse an affine constraint.
|
|
|
|
/// affine-constraint ::= affine-expr `>=` `0`
|
|
|
|
/// | affine-expr `==` `0`
|
|
|
|
///
|
|
|
|
/// isEq is set to true if the parsed constraint is an equality, false if it is
|
|
|
|
/// an inequality (greater than or equal).
|
|
|
|
///
|
|
|
|
AffineExpr *AffineParser::parseAffineConstraint(bool *isEq) {
|
|
|
|
AffineExpr *expr = parseAffineExpr();
|
|
|
|
if (!expr)
|
|
|
|
return nullptr;
|
|
|
|
|
|
|
|
if (consumeIf(Token::greater) && consumeIf(Token::equal) &&
|
|
|
|
getToken().is(Token::integer)) {
|
|
|
|
auto dim = getToken().getUnsignedIntegerValue();
|
|
|
|
if (dim.hasValue() && dim.getValue() == 0) {
|
|
|
|
consumeToken(Token::integer);
|
|
|
|
*isEq = false;
|
|
|
|
return expr;
|
|
|
|
}
|
|
|
|
return (emitError("expected '0' after '>='"), nullptr);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (consumeIf(Token::equal) && consumeIf(Token::equal) &&
|
|
|
|
getToken().is(Token::integer)) {
|
|
|
|
auto dim = getToken().getUnsignedIntegerValue();
|
|
|
|
if (dim.hasValue() && dim.getValue() == 0) {
|
|
|
|
consumeToken(Token::integer);
|
|
|
|
*isEq = true;
|
|
|
|
return expr;
|
|
|
|
}
|
|
|
|
return (emitError("expected '0' after '=='"), nullptr);
|
|
|
|
}
|
|
|
|
|
|
|
|
return (emitError("expected '== 0' or '>= 0' at end of affine constraint"),
|
|
|
|
nullptr);
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Parse an integer set definition.
|
|
|
|
/// integer-set-inline
|
|
|
|
/// ::= dim-and-symbol-id-lists `:` affine-constraint-conjunction
|
|
|
|
/// affine-constraint-conjunction ::= /*empty*/
|
|
|
|
/// | affine-constraint (`,` affine-constraint)*
|
|
|
|
///
|
|
|
|
IntegerSet *AffineParser::parseIntegerSetInline() {
|
|
|
|
unsigned numDims = 0, numSymbols = 0;
|
|
|
|
|
|
|
|
// List of dimensional identifiers.
|
|
|
|
if (parseDimIdList(numDims))
|
|
|
|
return nullptr;
|
|
|
|
|
|
|
|
// Symbols are optional.
|
|
|
|
if (getToken().is(Token::l_square)) {
|
|
|
|
if (parseSymbolIdList(numSymbols))
|
|
|
|
return nullptr;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (parseToken(Token::colon, "expected ':' or '['") ||
|
|
|
|
parseToken(Token::l_paren,
|
|
|
|
"expected '(' at start of integer set constraint list"))
|
|
|
|
return nullptr;
|
|
|
|
|
|
|
|
SmallVector<AffineExpr *, 4> constraints;
|
|
|
|
SmallVector<bool, 4> isEqs;
|
|
|
|
auto parseElt = [&]() -> ParseResult {
|
|
|
|
bool isEq;
|
|
|
|
auto *elt = parseAffineConstraint(&isEq);
|
|
|
|
ParseResult res = elt ? ParseSuccess : ParseFailure;
|
|
|
|
if (elt) {
|
|
|
|
constraints.push_back(elt);
|
|
|
|
isEqs.push_back(isEq);
|
|
|
|
}
|
|
|
|
return res;
|
|
|
|
};
|
|
|
|
|
|
|
|
// Parse a list of affine constraints (comma-separated) .
|
|
|
|
// Grammar: affine-constraint-conjunct ::= `(` affine-constraint (`,`
|
|
|
|
// affine-constraint)* `)
|
|
|
|
if (parseCommaSeparatedListUntil(Token::r_paren, parseElt, true))
|
|
|
|
return nullptr;
|
|
|
|
|
|
|
|
// Parsed a valid integer set.
|
|
|
|
return builder.getIntegerSet(numDims, numSymbols, constraints, isEqs);
|
|
|
|
}
|
|
|
|
|
|
|
|
IntegerSet *Parser::parseIntegerSetInline() {
|
|
|
|
return AffineParser(state).parseIntegerSetInline();
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Parse a reference to an integer set.
|
|
|
|
/// integer-set ::= integer-set-id | integer-set-inline
|
|
|
|
/// integer-set-id ::= `@@` suffix-id
|
|
|
|
///
|
|
|
|
IntegerSet *Parser::parseIntegerSetReference() {
|
|
|
|
if (getToken().is(Token::double_at_identifier)) {
|
|
|
|
// Parse integer set identifier and verify that it exists.
|
|
|
|
StringRef integerSetId = getTokenSpelling().drop_front(2);
|
|
|
|
if (getState().integerSetDefinitions.count(integerSetId) == 0)
|
|
|
|
return (emitError("undefined integer set id '" + integerSetId + "'"),
|
|
|
|
nullptr);
|
|
|
|
consumeToken(Token::double_at_identifier);
|
|
|
|
return getState().integerSetDefinitions[integerSetId];
|
|
|
|
}
|
|
|
|
// Try to parse an inline integer set definition.
|
|
|
|
return parseIntegerSetInline();
|
|
|
|
}
|
|
|
|
|
2018-07-04 08:51:28 +08:00
|
|
|
/// If statement.
|
|
|
|
///
|
2018-07-10 10:05:38 +08:00
|
|
|
/// ml-if-head ::= `if` ml-if-cond `{` ml-stmt* `}`
|
|
|
|
/// | ml-if-head `else` `if` ml-if-cond `{` ml-stmt* `}`
|
|
|
|
/// ml-if-stmt ::= ml-if-head
|
|
|
|
/// | ml-if-head `else` `{` ml-stmt* `}`
|
2018-07-04 08:51:28 +08:00
|
|
|
///
|
2018-07-17 02:47:09 +08:00
|
|
|
ParseResult MLFunctionParser::parseIfStmt() {
|
2018-08-24 05:32:25 +08:00
|
|
|
auto loc = getToken().getLoc();
|
2018-07-04 08:51:28 +08:00
|
|
|
consumeToken(Token::kw_if);
|
2018-08-08 05:24:38 +08:00
|
|
|
|
|
|
|
if (parseToken(Token::l_paren, "expected '('"))
|
2018-07-24 08:30:01 +08:00
|
|
|
return ParseFailure;
|
2018-07-04 08:51:28 +08:00
|
|
|
|
2018-08-08 05:24:38 +08:00
|
|
|
IntegerSet *condition = parseCondition();
|
|
|
|
if (!condition)
|
|
|
|
return ParseFailure;
|
2018-07-14 04:03:13 +08:00
|
|
|
|
2018-08-08 05:24:38 +08:00
|
|
|
if (parseToken(Token::r_paren, "expected ')'"))
|
2018-07-24 08:30:01 +08:00
|
|
|
return ParseFailure;
|
2018-07-14 04:03:13 +08:00
|
|
|
|
2018-08-24 05:32:25 +08:00
|
|
|
IfStmt *ifStmt = builder.createIf(getEncodedSourceLocation(loc), condition);
|
2018-08-09 02:14:57 +08:00
|
|
|
IfClause *thenClause = ifStmt->getThen();
|
2018-07-17 02:47:09 +08:00
|
|
|
|
2018-07-20 00:52:39 +08:00
|
|
|
// When parsing of an if statement body fails, the IR contains
|
|
|
|
// the if statement with the portion of the body that has been
|
|
|
|
// successfully parsed.
|
2018-07-17 02:47:09 +08:00
|
|
|
if (parseStmtBlock(thenClause))
|
|
|
|
return ParseFailure;
|
2018-06-29 08:02:32 +08:00
|
|
|
|
2018-07-14 04:03:13 +08:00
|
|
|
if (consumeIf(Token::kw_else)) {
|
2018-08-09 02:14:57 +08:00
|
|
|
auto *elseClause = ifStmt->createElse();
|
2018-07-17 02:47:09 +08:00
|
|
|
if (parseElseClause(elseClause))
|
|
|
|
return ParseFailure;
|
2018-07-04 08:51:28 +08:00
|
|
|
}
|
|
|
|
|
2018-07-31 06:18:10 +08:00
|
|
|
// Reset insertion point to the current block.
|
2018-08-09 02:14:57 +08:00
|
|
|
builder.setInsertionPointToEnd(ifStmt->getBlock());
|
2018-07-31 06:18:10 +08:00
|
|
|
|
2018-07-17 02:47:09 +08:00
|
|
|
return ParseSuccess;
|
2018-07-14 04:03:13 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
ParseResult MLFunctionParser::parseElseClause(IfClause *elseClause) {
|
|
|
|
if (getToken().is(Token::kw_if)) {
|
2018-08-09 02:14:57 +08:00
|
|
|
builder.setInsertionPointToEnd(elseClause);
|
2018-07-17 02:47:09 +08:00
|
|
|
return parseIfStmt();
|
2018-07-14 04:03:13 +08:00
|
|
|
}
|
|
|
|
|
2018-07-17 02:47:09 +08:00
|
|
|
return parseStmtBlock(elseClause);
|
|
|
|
}
|
|
|
|
|
|
|
|
///
|
|
|
|
/// Parse a list of statements ending with `return` or `}`
|
|
|
|
///
|
|
|
|
ParseResult MLFunctionParser::parseStatements(StmtBlock *block) {
|
2018-08-08 03:02:37 +08:00
|
|
|
auto createOpFunc = [&](const OperationState &state) -> Operation * {
|
|
|
|
return builder.createOperation(state);
|
2018-07-17 02:47:09 +08:00
|
|
|
};
|
|
|
|
|
2018-08-09 02:14:57 +08:00
|
|
|
builder.setInsertionPointToEnd(block);
|
2018-07-17 02:47:09 +08:00
|
|
|
|
2018-08-10 03:28:58 +08:00
|
|
|
// Parse statements till we see '}' or 'return'.
|
|
|
|
// Return statement is parsed separately to emit a more intuitive error
|
|
|
|
// when '}' is missing after the return statement.
|
|
|
|
while (getToken().isNot(Token::r_brace, Token::kw_return)) {
|
2018-07-17 02:47:09 +08:00
|
|
|
switch (getToken().getKind()) {
|
|
|
|
default:
|
|
|
|
if (parseOperation(createOpFunc))
|
|
|
|
return ParseFailure;
|
|
|
|
break;
|
|
|
|
case Token::kw_for:
|
|
|
|
if (parseForStmt())
|
|
|
|
return ParseFailure;
|
|
|
|
break;
|
|
|
|
case Token::kw_if:
|
|
|
|
if (parseIfStmt())
|
|
|
|
return ParseFailure;
|
|
|
|
break;
|
|
|
|
} // end switch
|
|
|
|
}
|
2018-07-14 04:03:13 +08:00
|
|
|
|
2018-08-10 03:28:58 +08:00
|
|
|
// Parse the return statement.
|
|
|
|
if (getToken().is(Token::kw_return))
|
|
|
|
if (parseOperation(createOpFunc))
|
|
|
|
return ParseFailure;
|
|
|
|
|
2018-07-14 04:03:13 +08:00
|
|
|
return ParseSuccess;
|
2018-07-04 08:51:28 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
///
|
|
|
|
/// Parse `{` ml-stmt* `}`
|
|
|
|
///
|
2018-07-14 04:03:13 +08:00
|
|
|
ParseResult MLFunctionParser::parseStmtBlock(StmtBlock *block) {
|
2018-07-24 08:30:01 +08:00
|
|
|
if (parseToken(Token::l_brace, "expected '{' before statement list") ||
|
|
|
|
parseStatements(block) ||
|
2018-08-10 03:28:58 +08:00
|
|
|
parseToken(Token::r_brace, "expected '}' after statement list"))
|
2018-07-17 02:47:09 +08:00
|
|
|
return ParseFailure;
|
|
|
|
|
2018-07-04 08:51:28 +08:00
|
|
|
return ParseSuccess;
|
2018-06-29 08:02:32 +08:00
|
|
|
}
|
2018-06-25 02:18:29 +08:00
|
|
|
|
2018-06-24 07:03:42 +08:00
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
// Top-level entity parsing.
|
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
|
2018-07-11 01:08:27 +08:00
|
|
|
namespace {
|
|
|
|
/// This parser handles entities that are only valid at the top level of the
|
|
|
|
/// file.
|
|
|
|
class ModuleParser : public Parser {
|
|
|
|
public:
|
|
|
|
explicit ModuleParser(ParserState &state) : Parser(state) {}
|
|
|
|
|
|
|
|
ParseResult parseModule();
|
|
|
|
|
|
|
|
private:
|
2018-08-20 12:17:22 +08:00
|
|
|
ParseResult finalizeModule();
|
|
|
|
|
2018-07-11 01:08:27 +08:00
|
|
|
ParseResult parseAffineMapDef();
|
2018-08-08 05:24:38 +08:00
|
|
|
ParseResult parseIntegerSetDef();
|
2018-07-11 01:08:27 +08:00
|
|
|
|
|
|
|
// Functions.
|
2018-07-20 00:52:39 +08:00
|
|
|
ParseResult parseMLArgumentList(SmallVectorImpl<Type *> &argTypes,
|
|
|
|
SmallVectorImpl<StringRef> &argNames);
|
|
|
|
ParseResult parseFunctionSignature(StringRef &name, FunctionType *&type,
|
|
|
|
SmallVectorImpl<StringRef> *argNames);
|
2018-07-11 01:08:27 +08:00
|
|
|
ParseResult parseExtFunc();
|
|
|
|
ParseResult parseCFGFunc();
|
|
|
|
ParseResult parseMLFunc();
|
|
|
|
};
|
|
|
|
} // end anonymous namespace
|
|
|
|
|
|
|
|
/// Affine map declaration.
|
|
|
|
///
|
|
|
|
/// affine-map-def ::= affine-map-id `=` affine-map-inline
|
|
|
|
///
|
|
|
|
ParseResult ModuleParser::parseAffineMapDef() {
|
|
|
|
assert(getToken().is(Token::hash_identifier));
|
|
|
|
|
|
|
|
StringRef affineMapId = getTokenSpelling().drop_front();
|
|
|
|
|
|
|
|
// Check for redefinitions.
|
2018-08-08 05:24:38 +08:00
|
|
|
auto **entry = &getState().affineMapDefinitions[affineMapId];
|
|
|
|
if (*entry)
|
2018-07-11 01:08:27 +08:00
|
|
|
return emitError("redefinition of affine map id '" + affineMapId + "'");
|
|
|
|
|
|
|
|
consumeToken(Token::hash_identifier);
|
|
|
|
|
|
|
|
// Parse the '='
|
2018-07-24 08:30:01 +08:00
|
|
|
if (parseToken(Token::equal,
|
|
|
|
"expected '=' in affine map outlined definition"))
|
|
|
|
return ParseFailure;
|
2018-07-11 01:08:27 +08:00
|
|
|
|
2018-08-08 05:24:38 +08:00
|
|
|
*entry = parseAffineMapInline();
|
|
|
|
if (!*entry)
|
|
|
|
return ParseFailure;
|
|
|
|
|
|
|
|
return ParseSuccess;
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Integer set declaration.
|
|
|
|
///
|
|
|
|
/// integer-set-decl ::= integer-set-id `=` integer-set-inline
|
|
|
|
///
|
|
|
|
ParseResult ModuleParser::parseIntegerSetDef() {
|
|
|
|
assert(getToken().is(Token::double_at_identifier));
|
|
|
|
|
|
|
|
StringRef integerSetId = getTokenSpelling().drop_front(2);
|
|
|
|
|
|
|
|
// Check for redefinitions (a default entry is created if one doesn't exist)
|
|
|
|
auto **entry = &getState().integerSetDefinitions[integerSetId];
|
|
|
|
if (*entry)
|
|
|
|
return emitError("redefinition of integer set id '" + integerSetId + "'");
|
|
|
|
|
|
|
|
consumeToken(Token::double_at_identifier);
|
|
|
|
|
|
|
|
// Parse the '='
|
|
|
|
if (parseToken(Token::equal,
|
|
|
|
"expected '=' in outlined integer set definition"))
|
|
|
|
return ParseFailure;
|
|
|
|
|
|
|
|
*entry = parseIntegerSetInline();
|
|
|
|
if (!*entry)
|
2018-07-11 01:08:27 +08:00
|
|
|
return ParseFailure;
|
|
|
|
|
|
|
|
return ParseSuccess;
|
|
|
|
}
|
|
|
|
|
2018-07-20 00:52:39 +08:00
|
|
|
/// Parse a (possibly empty) list of MLFunction arguments with types.
|
|
|
|
///
|
|
|
|
/// ml-argument ::= ssa-id `:` type
|
|
|
|
/// ml-argument-list ::= ml-argument (`,` ml-argument)* | /*empty*/
|
|
|
|
///
|
|
|
|
ParseResult
|
|
|
|
ModuleParser::parseMLArgumentList(SmallVectorImpl<Type *> &argTypes,
|
|
|
|
SmallVectorImpl<StringRef> &argNames) {
|
2018-07-24 08:30:01 +08:00
|
|
|
consumeToken(Token::l_paren);
|
|
|
|
|
2018-07-20 00:52:39 +08:00
|
|
|
auto parseElt = [&]() -> ParseResult {
|
|
|
|
// Parse argument name
|
|
|
|
if (getToken().isNot(Token::percent_identifier))
|
|
|
|
return emitError("expected SSA identifier");
|
|
|
|
|
2018-08-07 02:54:39 +08:00
|
|
|
StringRef name = getTokenSpelling();
|
2018-07-20 00:52:39 +08:00
|
|
|
consumeToken(Token::percent_identifier);
|
|
|
|
argNames.push_back(name);
|
|
|
|
|
2018-07-24 08:30:01 +08:00
|
|
|
if (parseToken(Token::colon, "expected ':'"))
|
|
|
|
return ParseFailure;
|
2018-07-20 00:52:39 +08:00
|
|
|
|
|
|
|
// Parse argument type
|
|
|
|
auto elt = parseType();
|
|
|
|
if (!elt)
|
|
|
|
return ParseFailure;
|
|
|
|
argTypes.push_back(elt);
|
|
|
|
|
|
|
|
return ParseSuccess;
|
|
|
|
};
|
|
|
|
|
2018-07-22 05:32:09 +08:00
|
|
|
return parseCommaSeparatedListUntil(Token::r_paren, parseElt);
|
2018-07-20 00:52:39 +08:00
|
|
|
}
|
|
|
|
|
2018-07-11 01:08:27 +08:00
|
|
|
/// Parse a function signature, starting with a name and including the parameter
|
|
|
|
/// list.
|
|
|
|
///
|
2018-07-20 00:52:39 +08:00
|
|
|
/// argument-list ::= type (`,` type)* | /*empty*/ | ml-argument-list
|
2018-07-11 01:08:27 +08:00
|
|
|
/// function-signature ::= function-id `(` argument-list `)` (`->` type-list)?
|
|
|
|
///
|
2018-07-20 00:52:39 +08:00
|
|
|
ParseResult
|
|
|
|
ModuleParser::parseFunctionSignature(StringRef &name, FunctionType *&type,
|
|
|
|
SmallVectorImpl<StringRef> *argNames) {
|
2018-07-11 01:08:27 +08:00
|
|
|
if (getToken().isNot(Token::at_identifier))
|
|
|
|
return emitError("expected a function identifier like '@foo'");
|
|
|
|
|
|
|
|
name = getTokenSpelling().drop_front();
|
|
|
|
consumeToken(Token::at_identifier);
|
|
|
|
|
|
|
|
if (getToken().isNot(Token::l_paren))
|
|
|
|
return emitError("expected '(' in function signature");
|
|
|
|
|
2018-07-20 00:52:39 +08:00
|
|
|
SmallVector<Type *, 4> argTypes;
|
|
|
|
ParseResult parseResult;
|
|
|
|
|
|
|
|
if (argNames)
|
|
|
|
parseResult = parseMLArgumentList(argTypes, *argNames);
|
|
|
|
else
|
|
|
|
parseResult = parseTypeList(argTypes);
|
|
|
|
|
|
|
|
if (parseResult)
|
2018-07-11 01:08:27 +08:00
|
|
|
return ParseFailure;
|
|
|
|
|
|
|
|
// Parse the return type if present.
|
|
|
|
SmallVector<Type *, 4> results;
|
|
|
|
if (consumeIf(Token::arrow)) {
|
|
|
|
if (parseTypeList(results))
|
|
|
|
return ParseFailure;
|
|
|
|
}
|
2018-07-20 00:52:39 +08:00
|
|
|
type = builder.getFunctionType(argTypes, results);
|
2018-07-11 01:08:27 +08:00
|
|
|
return ParseSuccess;
|
|
|
|
}
|
|
|
|
|
|
|
|
/// External function declarations.
|
|
|
|
///
|
|
|
|
/// ext-func ::= `extfunc` function-signature
|
|
|
|
///
|
|
|
|
ParseResult ModuleParser::parseExtFunc() {
|
|
|
|
consumeToken(Token::kw_extfunc);
|
2018-08-18 07:49:42 +08:00
|
|
|
auto loc = getToken().getLoc();
|
2018-07-11 01:08:27 +08:00
|
|
|
|
|
|
|
StringRef name;
|
|
|
|
FunctionType *type = nullptr;
|
2018-07-20 00:52:39 +08:00
|
|
|
if (parseFunctionSignature(name, type, /*arguments*/ nullptr))
|
2018-07-11 01:08:27 +08:00
|
|
|
return ParseFailure;
|
|
|
|
|
|
|
|
// Okay, the external function definition was parsed correctly.
|
2018-08-18 07:49:42 +08:00
|
|
|
auto *function = new ExtFunction(name, type);
|
|
|
|
getModule()->getFunctions().push_back(function);
|
|
|
|
|
|
|
|
// Verify no name collision / redefinition.
|
2018-08-20 12:17:22 +08:00
|
|
|
if (function->getName() != name)
|
2018-08-18 07:49:42 +08:00
|
|
|
return emitError(loc,
|
|
|
|
"redefinition of function named '" + name.str() + "'");
|
|
|
|
|
2018-07-11 01:08:27 +08:00
|
|
|
return ParseSuccess;
|
|
|
|
}
|
|
|
|
|
|
|
|
/// CFG function declarations.
|
|
|
|
///
|
|
|
|
/// cfg-func ::= `cfgfunc` function-signature `{` basic-block+ `}`
|
|
|
|
///
|
|
|
|
ParseResult ModuleParser::parseCFGFunc() {
|
|
|
|
consumeToken(Token::kw_cfgfunc);
|
2018-08-18 07:49:42 +08:00
|
|
|
auto loc = getToken().getLoc();
|
2018-07-11 01:08:27 +08:00
|
|
|
|
|
|
|
StringRef name;
|
|
|
|
FunctionType *type = nullptr;
|
2018-07-20 00:52:39 +08:00
|
|
|
if (parseFunctionSignature(name, type, /*arguments*/ nullptr))
|
2018-07-11 01:08:27 +08:00
|
|
|
return ParseFailure;
|
|
|
|
|
|
|
|
// Okay, the CFG function signature was parsed correctly, create the function.
|
2018-08-18 07:49:42 +08:00
|
|
|
auto *function = new CFGFunction(name, type);
|
|
|
|
getModule()->getFunctions().push_back(function);
|
|
|
|
|
|
|
|
// Verify no name collision / redefinition.
|
2018-08-20 12:17:22 +08:00
|
|
|
if (function->getName() != name)
|
2018-08-18 07:49:42 +08:00
|
|
|
return emitError(loc,
|
|
|
|
"redefinition of function named '" + name.str() + "'");
|
2018-07-11 01:08:27 +08:00
|
|
|
|
|
|
|
return CFGFunctionParser(getState(), function).parseFunctionBody();
|
|
|
|
}
|
|
|
|
|
|
|
|
/// ML function declarations.
|
|
|
|
///
|
|
|
|
/// ml-func ::= `mlfunc` ml-func-signature `{` ml-stmt* ml-return-stmt `}`
|
|
|
|
///
|
|
|
|
ParseResult ModuleParser::parseMLFunc() {
|
|
|
|
consumeToken(Token::kw_mlfunc);
|
|
|
|
|
|
|
|
StringRef name;
|
|
|
|
FunctionType *type = nullptr;
|
2018-07-20 00:52:39 +08:00
|
|
|
SmallVector<StringRef, 4> argNames;
|
|
|
|
|
2018-08-07 02:54:39 +08:00
|
|
|
auto loc = getToken().getLoc();
|
2018-07-20 00:52:39 +08:00
|
|
|
if (parseFunctionSignature(name, type, &argNames))
|
2018-07-11 01:08:27 +08:00
|
|
|
return ParseFailure;
|
|
|
|
|
|
|
|
// Okay, the ML function signature was parsed correctly, create the function.
|
2018-08-18 07:49:42 +08:00
|
|
|
auto *function = MLFunction::create(name, type);
|
|
|
|
getModule()->getFunctions().push_back(function);
|
|
|
|
|
|
|
|
// Verify no name collision / redefinition.
|
2018-08-20 12:17:22 +08:00
|
|
|
if (function->getName() != name)
|
2018-08-18 07:49:42 +08:00
|
|
|
return emitError(loc,
|
|
|
|
"redefinition of function named '" + name.str() + "'");
|
2018-08-07 02:54:39 +08:00
|
|
|
|
|
|
|
// Create the parser.
|
|
|
|
auto parser = MLFunctionParser(getState(), function);
|
|
|
|
|
|
|
|
// Add definitions of the function arguments.
|
|
|
|
for (unsigned i = 0, e = function->getNumArguments(); i != e; ++i) {
|
|
|
|
if (parser.addDefinition({argNames[i], 0, loc}, function->getArgument(i)))
|
|
|
|
return ParseFailure;
|
|
|
|
}
|
2018-07-11 01:08:27 +08:00
|
|
|
|
2018-08-07 02:54:39 +08:00
|
|
|
return parser.parseFunctionBody();
|
2018-07-11 01:08:27 +08:00
|
|
|
}
|
|
|
|
|
2018-08-21 23:42:19 +08:00
|
|
|
/// Given an attribute that could refer to a function attribute in the remapping
|
|
|
|
/// table, walk it and rewrite it to use the mapped function. If it doesn't
|
|
|
|
/// refer to anything in the table, then it is returned unmodified.
|
|
|
|
static Attribute *
|
|
|
|
remapFunctionAttrs(Attribute *input,
|
|
|
|
DenseMap<FunctionAttr *, FunctionAttr *> &remappingTable,
|
|
|
|
MLIRContext *context) {
|
|
|
|
// Most attributes are trivially unrelated to function attributes, skip them
|
|
|
|
// rapidly.
|
|
|
|
if (!input->isOrContainsFunction())
|
|
|
|
return input;
|
|
|
|
|
|
|
|
// If we have a function attribute, remap it.
|
|
|
|
if (auto *fnAttr = dyn_cast<FunctionAttr>(input)) {
|
|
|
|
auto it = remappingTable.find(fnAttr);
|
|
|
|
return it != remappingTable.end() ? it->second : input;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Otherwise, we must have an array attribute, remap the elements.
|
|
|
|
auto *arrayAttr = cast<ArrayAttr>(input);
|
|
|
|
SmallVector<Attribute *, 8> remappedElts;
|
|
|
|
bool anyChange = false;
|
|
|
|
for (auto *elt : arrayAttr->getValue()) {
|
|
|
|
auto *newElt = remapFunctionAttrs(elt, remappingTable, context);
|
|
|
|
remappedElts.push_back(newElt);
|
|
|
|
anyChange |= (elt != newElt);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!anyChange)
|
|
|
|
return input;
|
|
|
|
|
|
|
|
return ArrayAttr::get(remappedElts, context);
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Remap function attributes to resolve forward references to their actual
|
|
|
|
/// definition.
|
|
|
|
static void remapFunctionAttrsInOperation(
|
|
|
|
Operation *op, DenseMap<FunctionAttr *, FunctionAttr *> &remappingTable) {
|
|
|
|
for (auto attr : op->getAttrs()) {
|
|
|
|
// Do the remapping, if we got the same thing back, then it must contain
|
|
|
|
// functions that aren't getting remapped.
|
|
|
|
auto *newVal =
|
|
|
|
remapFunctionAttrs(attr.second, remappingTable, op->getContext());
|
|
|
|
if (newVal == attr.second)
|
|
|
|
continue;
|
|
|
|
|
|
|
|
// Otherwise, replace the existing attribute with the new one. It is safe
|
|
|
|
// to mutate the attribute list while we walk it because underlying
|
|
|
|
// attribute lists are uniqued and immortal.
|
|
|
|
op->setAttr(attr.first, newVal);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-08-20 12:17:22 +08:00
|
|
|
/// Finish the end of module parsing - when the result is valid, do final
|
|
|
|
/// checking.
|
|
|
|
ParseResult ModuleParser::finalizeModule() {
|
|
|
|
|
2018-08-21 23:42:19 +08:00
|
|
|
// Resolve all forward references, building a remapping table of attributes.
|
|
|
|
DenseMap<FunctionAttr *, FunctionAttr *> remappingTable;
|
2018-08-20 12:17:22 +08:00
|
|
|
for (auto forwardRef : getState().functionForwardRefs) {
|
|
|
|
auto name = forwardRef.first;
|
|
|
|
|
|
|
|
// Resolve the reference.
|
|
|
|
auto *resolvedFunction = getModule()->getNamedFunction(name);
|
|
|
|
if (!resolvedFunction)
|
|
|
|
return emitError(forwardRef.second.second,
|
|
|
|
"reference to undefined function '" + name.str() + "'");
|
|
|
|
|
2018-08-21 23:42:19 +08:00
|
|
|
remappingTable[builder.getFunctionAttr(forwardRef.second.first)] =
|
|
|
|
builder.getFunctionAttr(resolvedFunction);
|
|
|
|
}
|
|
|
|
|
|
|
|
// If there was nothing to remap, then we're done.
|
|
|
|
if (remappingTable.empty())
|
|
|
|
return ParseSuccess;
|
|
|
|
|
|
|
|
// Otherwise, walk the entire module replacing uses of one attribute set with
|
|
|
|
// the correct ones.
|
|
|
|
for (auto &fn : *getModule()) {
|
|
|
|
if (auto *cfgFn = dyn_cast<CFGFunction>(&fn)) {
|
|
|
|
for (auto &bb : *cfgFn) {
|
|
|
|
for (auto &inst : bb) {
|
|
|
|
remapFunctionAttrsInOperation(&inst, remappingTable);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Otherwise, look at MLFunctions. We ignore ExtFunctions.
|
|
|
|
auto *mlFn = dyn_cast<MLFunction>(&fn);
|
|
|
|
if (!mlFn)
|
|
|
|
continue;
|
|
|
|
|
|
|
|
struct MLFnWalker : public StmtWalker<MLFnWalker> {
|
|
|
|
MLFnWalker(DenseMap<FunctionAttr *, FunctionAttr *> &remappingTable)
|
|
|
|
: remappingTable(remappingTable) {}
|
|
|
|
void visitOperationStmt(OperationStmt *opStmt) {
|
|
|
|
remapFunctionAttrsInOperation(opStmt, remappingTable);
|
|
|
|
}
|
|
|
|
|
|
|
|
DenseMap<FunctionAttr *, FunctionAttr *> &remappingTable;
|
|
|
|
};
|
|
|
|
|
|
|
|
MLFnWalker(remappingTable).walk(mlFn);
|
2018-08-20 12:17:22 +08:00
|
|
|
}
|
|
|
|
|
2018-08-21 23:42:19 +08:00
|
|
|
// Now that all references to the forward definition placeholders are
|
|
|
|
// resolved, we can deallocate the placeholders.
|
|
|
|
for (auto forwardRef : getState().functionForwardRefs)
|
|
|
|
forwardRef.second.first->destroy();
|
2018-08-20 12:17:22 +08:00
|
|
|
return ParseSuccess;
|
|
|
|
}
|
|
|
|
|
2018-06-23 01:39:19 +08:00
|
|
|
/// This is the top-level module parser.
|
2018-07-11 01:08:27 +08:00
|
|
|
ParseResult ModuleParser::parseModule() {
|
2018-06-23 01:39:19 +08:00
|
|
|
while (1) {
|
2018-07-10 10:05:38 +08:00
|
|
|
switch (getToken().getKind()) {
|
2018-06-23 01:39:19 +08:00
|
|
|
default:
|
|
|
|
emitError("expected a top level entity");
|
2018-07-11 01:08:27 +08:00
|
|
|
return ParseFailure;
|
2018-06-23 01:39:19 +08:00
|
|
|
|
2018-07-04 11:16:08 +08:00
|
|
|
// If we got to the end of the file, then we're done.
|
2018-06-23 01:39:19 +08:00
|
|
|
case Token::eof:
|
2018-08-20 12:17:22 +08:00
|
|
|
return finalizeModule();
|
2018-06-23 01:39:19 +08:00
|
|
|
|
|
|
|
// If we got an error token, then the lexer already emitted an error, just
|
|
|
|
// stop. Someday we could introduce error recovery if there was demand for
|
|
|
|
// it.
|
|
|
|
case Token::error:
|
2018-07-11 01:08:27 +08:00
|
|
|
return ParseFailure;
|
2018-06-23 01:39:19 +08:00
|
|
|
|
2018-07-11 01:08:27 +08:00
|
|
|
case Token::hash_identifier:
|
|
|
|
if (parseAffineMapDef())
|
|
|
|
return ParseFailure;
|
2018-06-24 07:03:42 +08:00
|
|
|
break;
|
|
|
|
|
2018-08-08 05:24:38 +08:00
|
|
|
case Token::double_at_identifier:
|
|
|
|
if (parseIntegerSetDef())
|
|
|
|
return ParseFailure;
|
|
|
|
break;
|
|
|
|
|
2018-07-11 01:08:27 +08:00
|
|
|
case Token::kw_extfunc:
|
|
|
|
if (parseExtFunc())
|
|
|
|
return ParseFailure;
|
2018-06-23 01:39:19 +08:00
|
|
|
break;
|
2018-07-04 11:16:08 +08:00
|
|
|
|
2018-07-11 01:08:27 +08:00
|
|
|
case Token::kw_cfgfunc:
|
|
|
|
if (parseCFGFunc())
|
|
|
|
return ParseFailure;
|
2018-06-28 02:03:08 +08:00
|
|
|
break;
|
2018-06-23 01:39:19 +08:00
|
|
|
|
2018-06-29 08:02:32 +08:00
|
|
|
case Token::kw_mlfunc:
|
2018-07-11 01:08:27 +08:00
|
|
|
if (parseMLFunc())
|
|
|
|
return ParseFailure;
|
2018-06-29 08:02:32 +08:00
|
|
|
break;
|
2018-06-23 01:39:19 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
|
2018-07-04 04:24:09 +08:00
|
|
|
void mlir::defaultErrorReporter(const llvm::SMDiagnostic &error) {
|
|
|
|
const auto &sourceMgr = *error.getSourceMgr();
|
|
|
|
sourceMgr.PrintMessage(error.getLoc(), error.getKind(), error.getMessage());
|
|
|
|
}
|
|
|
|
|
2018-06-23 01:39:19 +08:00
|
|
|
/// This parses the file specified by the indicated SourceMgr and returns an
|
|
|
|
/// MLIR module if it was valid. If not, it emits diagnostics and returns null.
|
2018-06-25 10:17:35 +08:00
|
|
|
Module *mlir::parseSourceFile(llvm::SourceMgr &sourceMgr, MLIRContext *context,
|
2018-07-04 04:24:09 +08:00
|
|
|
SMDiagnosticHandlerTy errorReporter) {
|
2018-08-21 23:42:19 +08:00
|
|
|
if (!errorReporter)
|
|
|
|
errorReporter = defaultErrorReporter;
|
|
|
|
|
|
|
|
// We are going to replace the context's handler and redirect it to use the
|
|
|
|
// error reporter. Save the existing handler and reinstate it when we're
|
|
|
|
// done.
|
|
|
|
auto existingContextHandler = context->getDiagnosticHandler();
|
|
|
|
|
|
|
|
// Install a new handler that uses the error reporter.
|
|
|
|
context->registerDiagnosticHandler([&](Attribute *location, StringRef message,
|
|
|
|
MLIRContext::DiagnosticKind kind) {
|
|
|
|
auto offset = cast<IntegerAttr>(location)->getValue();
|
|
|
|
auto *mainBuffer = sourceMgr.getMemoryBuffer(sourceMgr.getMainFileID());
|
|
|
|
auto ptr = mainBuffer->getBufferStart() + offset;
|
|
|
|
SourceMgr::DiagKind diagKind;
|
|
|
|
switch (kind) {
|
|
|
|
case MLIRContext::DiagnosticKind::Error:
|
|
|
|
diagKind = SourceMgr::DK_Error;
|
|
|
|
break;
|
|
|
|
case MLIRContext::DiagnosticKind::Warning:
|
|
|
|
diagKind = SourceMgr::DK_Warning;
|
|
|
|
break;
|
|
|
|
case MLIRContext::DiagnosticKind::Note:
|
|
|
|
diagKind = SourceMgr::DK_Note;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
errorReporter(
|
|
|
|
sourceMgr.GetMessage(SMLoc::getFromPointer(ptr), diagKind, message));
|
|
|
|
});
|
|
|
|
|
2018-07-11 01:08:27 +08:00
|
|
|
// This is the result module we are parsing into.
|
|
|
|
std::unique_ptr<Module> module(new Module(context));
|
|
|
|
|
2018-08-21 23:42:19 +08:00
|
|
|
ParserState state(sourceMgr, module.get(), errorReporter);
|
|
|
|
if (ModuleParser(state).parseModule()) {
|
|
|
|
context->registerDiagnosticHandler(existingContextHandler);
|
2018-07-11 01:08:27 +08:00
|
|
|
return nullptr;
|
2018-08-21 23:42:19 +08:00
|
|
|
}
|
2018-07-07 01:46:19 +08:00
|
|
|
|
|
|
|
// Make sure the parse module has no other structural problems detected by the
|
|
|
|
// verifier.
|
2018-08-21 23:42:19 +08:00
|
|
|
std::string errorResult;
|
|
|
|
module->verify(&errorResult);
|
|
|
|
|
|
|
|
// We don't have location information for general verifier errors, so emit the
|
|
|
|
// error on the first line.
|
|
|
|
if (!errorResult.empty()) {
|
|
|
|
auto *mainBuffer = sourceMgr.getMemoryBuffer(sourceMgr.getMainFileID());
|
|
|
|
errorReporter(sourceMgr.GetMessage(
|
|
|
|
SMLoc::getFromPointer(mainBuffer->getBufferStart()),
|
|
|
|
SourceMgr::DK_Error, errorResult));
|
|
|
|
return nullptr;
|
|
|
|
}
|
|
|
|
|
|
|
|
context->registerDiagnosticHandler(existingContextHandler);
|
2018-07-11 01:08:27 +08:00
|
|
|
return module.release();
|
2018-06-23 01:39:19 +08:00
|
|
|
}
|