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-10-11 05:23:30 +08:00
|
|
|
#include "mlir/IR/BuiltinOps.h"
|
2018-08-29 06:26:20 +08:00
|
|
|
#include "mlir/IR/IntegerSet.h"
|
2018-08-28 12:05:16 +08:00
|
|
|
#include "mlir/IR/Location.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"
|
2019-01-04 06:29:52 +08:00
|
|
|
#include "mlir/IR/StandardTypes.h"
|
2018-10-19 04:54:44 +08:00
|
|
|
#include "mlir/Support/STLExtras.h"
|
2018-11-15 06:26:47 +08:00
|
|
|
#include "mlir/Transforms/Utils.h"
|
2018-10-19 04:54:44 +08:00
|
|
|
#include "llvm/ADT/APInt.h"
|
2018-08-03 16:54:46 +08:00
|
|
|
#include "llvm/ADT/DenseMap.h"
|
2018-09-03 22:38:31 +08:00
|
|
|
#include "llvm/Support/MemoryBuffer.h"
|
2018-08-22 08:55:22 +08:00
|
|
|
#include "llvm/Support/PrettyStackTrace.h"
|
2018-09-03 22:38:31 +08:00
|
|
|
#include "llvm/Support/SMLoc.h"
|
2018-08-03 16:54:46 +08:00
|
|
|
#include "llvm/Support/SourceMgr.h"
|
2018-10-24 04:44:04 +08:00
|
|
|
#include <algorithm>
|
2018-06-23 01:39:19 +08:00
|
|
|
using namespace mlir;
|
2018-09-03 22:38:31 +08:00
|
|
|
using llvm::MemoryBuffer;
|
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-09-09 09:37:27 +08:00
|
|
|
ParserState(const llvm::SourceMgr &sourceMgr, Module *module)
|
2018-09-03 13:01:45 +08:00
|
|
|
: context(module->getContext()), module(module), lex(sourceMgr, context),
|
2018-10-22 10:49:31 +08:00
|
|
|
curToken(lex.lexToken()) {}
|
2018-07-11 01:08:27 +08:00
|
|
|
|
2018-10-10 05:40:41 +08:00
|
|
|
~ParserState() {
|
|
|
|
// Destroy the forward references upon error.
|
|
|
|
for (auto forwardRef : functionForwardRefs)
|
2018-12-28 03:07:34 +08:00
|
|
|
delete forwardRef.second;
|
2018-10-10 05:40:41 +08:00
|
|
|
functionForwardRefs.clear();
|
|
|
|
}
|
|
|
|
|
2018-07-11 01:08:27 +08:00
|
|
|
// A map from affine map identifier to AffineMap.
|
2018-10-10 07:39:24 +08:00
|
|
|
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.
|
2018-10-11 00:45:59 +08:00
|
|
|
llvm::StringMap<IntegerSet> integerSetDefinitions;
|
2018-06-23 01:39:19 +08:00
|
|
|
|
2019-01-08 10:42:04 +08:00
|
|
|
// A map from type alias identifier to Type.
|
|
|
|
llvm::StringMap<Type> typeAliasDefinitions;
|
|
|
|
|
2018-08-20 12:17:22 +08:00
|
|
|
// This keeps track of all forward references to functions along with the
|
2018-09-08 00:08:13 +08:00
|
|
|
// temporary function used to represent them.
|
|
|
|
llvm::DenseMap<Identifier, Function *> functionForwardRefs;
|
2018-08-20 12:17:22 +08:00
|
|
|
|
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-07-10 10:05:38 +08:00
|
|
|
};
|
|
|
|
} // end anonymous namespace
|
|
|
|
|
|
|
|
namespace {
|
|
|
|
|
|
|
|
/// 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-09-09 09:37:27 +08:00
|
|
|
const 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.
|
2018-11-09 04:28:35 +08:00
|
|
|
Location getEncodedSourceLocation(llvm::SMLoc loc) {
|
2018-09-03 13:01:45 +08:00
|
|
|
return state.lex.getEncodedSourceLocation(loc);
|
|
|
|
}
|
2018-08-24 05:32:25 +08:00
|
|
|
|
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-10-31 05:59:22 +08:00
|
|
|
VectorType parseVectorType();
|
2018-09-14 01:43:35 +08:00
|
|
|
ParseResult parseXInDimensionList();
|
2019-02-08 00:36:50 +08:00
|
|
|
ParseResult parseDimensionListRanked(SmallVectorImpl<int64_t> &dimensions,
|
|
|
|
bool allowDynamic);
|
2019-01-08 10:42:04 +08:00
|
|
|
Type parseExtendedType();
|
2018-10-31 05:59:22 +08:00
|
|
|
Type parseTensorType();
|
|
|
|
Type parseMemRefType();
|
|
|
|
Type parseFunctionType();
|
2019-02-06 03:47:02 +08:00
|
|
|
Type parseNonFunctionType();
|
2018-10-31 05:59:22 +08:00
|
|
|
Type parseType();
|
|
|
|
ParseResult parseTypeListNoParens(SmallVectorImpl<Type> &elements);
|
2019-02-06 03:47:02 +08:00
|
|
|
ParseResult parseTypeListParens(SmallVectorImpl<Type> &elements);
|
|
|
|
ParseResult parseFunctionResultTypes(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,
|
2018-10-31 05:59:22 +08:00
|
|
|
FunctionType type);
|
2018-11-16 09:53:51 +08:00
|
|
|
Attribute parseAttribute(Type type = {});
|
Add support to constant sparse tensor / vector attribute
The SparseElementsAttr uses (COO) Coordinate List encoding to represents a
sparse tensor / vector. Specifically, the coordinates and values are stored as
two dense elements attributes. The first dense elements attribute is a 2-D
attribute with shape [N, ndims], which contains the indices of the elements
with nonzero values in the constant vector/tensor. The second elements
attribute is a 1-D attribute list with shape [N], which supplies the values for
each element in the first elements attribute. ndims is the rank of the
vector/tensor and N is the total nonzero elements.
The syntax is:
`sparse<` (tensor-type | vector-type)`, ` indices-attribute-list, values-attribute-list `>`
Example: a sparse tensor
sparse<vector<3x4xi32>, [[0, 0], [1, 2]], [1, 2]> represents the dense tensor
[[1, 0, 0, 0]
[0, 0, 2, 0]
[0, 0, 0, 0]]
PiperOrigin-RevId: 217764319
2018-10-19 05:02:20 +08:00
|
|
|
|
2018-07-05 11:45:39 +08:00
|
|
|
ParseResult parseAttributeDict(SmallVectorImpl<NamedAttribute> &attributes);
|
|
|
|
|
2018-07-04 11:16:08 +08:00
|
|
|
// Polyhedral structures.
|
2018-10-10 07:39:24 +08:00
|
|
|
AffineMap parseAffineMapReference();
|
2018-10-11 00:45:59 +08:00
|
|
|
IntegerSet parseIntegerSetReference();
|
2019-01-27 02:41:17 +08:00
|
|
|
ParseResult parseAffineMapOrIntegerSetReference(AffineMap &map,
|
|
|
|
IntegerSet &set);
|
2018-10-31 05:59:22 +08:00
|
|
|
DenseElementsAttr parseDenseElementsAttr(VectorOrTensorType type);
|
2019-02-08 02:13:50 +08:00
|
|
|
DenseElementsAttr parseDenseElementsAttrAsTensor(Type eltType);
|
2018-10-31 05:59:22 +08:00
|
|
|
VectorOrTensorType parseVectorOrTensorType();
|
2018-06-28 02:03:08 +08:00
|
|
|
|
2019-01-24 05:11:23 +08:00
|
|
|
// Location Parsing.
|
|
|
|
|
|
|
|
/// Trailing locations.
|
|
|
|
///
|
|
|
|
/// trailing-location ::= location?
|
|
|
|
///
|
|
|
|
template <typename Owner>
|
|
|
|
ParseResult parseOptionalTrailingLocation(Owner *owner) {
|
|
|
|
// If there is a 'loc' we parse a trailing location.
|
|
|
|
if (!getToken().is(Token::kw_loc))
|
|
|
|
return ParseSuccess;
|
|
|
|
|
|
|
|
// Parse the location.
|
|
|
|
llvm::Optional<Location> directLoc;
|
|
|
|
if (parseLocation(&directLoc))
|
|
|
|
return ParseFailure;
|
|
|
|
owner->setLoc(*directLoc);
|
|
|
|
return ParseSuccess;
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Parse an inline location.
|
|
|
|
ParseResult parseLocation(llvm::Optional<Location> *loc);
|
|
|
|
|
|
|
|
/// Parse a raw location instance.
|
|
|
|
ParseResult parseLocationInstance(llvm::Optional<Location> *loc);
|
|
|
|
|
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-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-12-08 01:30:25 +08:00
|
|
|
getContext()->emitError(getEncodedSourceLocation(loc), 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
|
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
|
2019-02-06 03:47:02 +08:00
|
|
|
/// Parse any type except the function 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
|
|
|
///
|
2019-02-06 03:47:02 +08:00
|
|
|
/// non-function-type ::= integer-type
|
|
|
|
/// | index-type
|
|
|
|
/// | float-type
|
|
|
|
/// | extended-type
|
|
|
|
/// | vector-type
|
|
|
|
/// | tensor-type
|
|
|
|
/// | memref-type
|
2018-06-23 06:52:02 +08:00
|
|
|
///
|
Enable arithmetics for index types.
Arithmetic and comparison instructions are necessary to implement, e.g.,
control flow when lowering MLFunctions to CFGFunctions. (While it is possible
to replace some of the arithmetics by affine_apply instructions for loop
bounds, it is still necessary for loop bounds checking, steps, if-conditions,
non-trivial memref subscripts, etc.) Furthermore, working with indirect
accesses in, e.g., lookup tables for large embeddings, may require operating on
tensors of indexes. For example, the equivalents to C code "LUT[Index[i]]" or
"ResultIndex[i] = i + j" where i, j are loop induction variables require the
arithmetics on indices as well as the possibility to operate on tensors
thereof. Allow arithmetic and comparison operations to apply to index types by
declaring them integer-like. Allow tensors whose element type is index for
indirection purposes.
The absence of vectors with "index" element type is explicitly tested, but the
only justification for this restriction in the CL introducing the test is
"because we don't need them". Do NOT enable vectors of index types, although
it makes vector and tensor types inconsistent with respect to allowed element
types.
PiperOrigin-RevId: 220614055
2018-11-08 20:04:32 +08:00
|
|
|
/// index-type ::= `index`
|
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`
|
2018-06-23 06:52:02 +08:00
|
|
|
///
|
2019-02-06 03:47:02 +08:00
|
|
|
Type Parser::parseNonFunctionType() {
|
2018-07-10 10:05:38 +08:00
|
|
|
switch (getToken().getKind()) {
|
2018-06-23 13:03:48 +08:00
|
|
|
default:
|
2019-02-06 03:47:02 +08:00
|
|
|
return (emitError("expected non-function 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();
|
|
|
|
// integer-type
|
|
|
|
case Token::inttype: {
|
|
|
|
auto width = getToken().getIntTypeBitwidth();
|
|
|
|
if (!width.hasValue())
|
|
|
|
return (emitError("invalid integer width"), nullptr);
|
2018-11-13 07:12:09 +08:00
|
|
|
auto loc = getEncodedSourceLocation(getToken().getLoc());
|
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
|
|
|
consumeToken(Token::inttype);
|
2018-11-13 07:12:09 +08:00
|
|
|
return IntegerType::getChecked(width.getValue(), builder.getContext(), loc);
|
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
|
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
|
|
|
|
Enable arithmetics for index types.
Arithmetic and comparison instructions are necessary to implement, e.g.,
control flow when lowering MLFunctions to CFGFunctions. (While it is possible
to replace some of the arithmetics by affine_apply instructions for loop
bounds, it is still necessary for loop bounds checking, steps, if-conditions,
non-trivial memref subscripts, etc.) Furthermore, working with indirect
accesses in, e.g., lookup tables for large embeddings, may require operating on
tensors of indexes. For example, the equivalents to C code "LUT[Index[i]]" or
"ResultIndex[i] = i + j" where i, j are loop induction variables require the
arithmetics on indices as well as the possibility to operate on tensors
thereof. Allow arithmetic and comparison operations to apply to index types by
declaring them integer-like. Allow tensors whose element type is index for
indirection purposes.
The absence of vectors with "index" element type is explicitly tested, but the
only justification for this restriction in the CL introducing the test is
"because we don't need them". Do NOT enable vectors of index types, although
it makes vector and tensor types inconsistent with respect to allowed element
types.
PiperOrigin-RevId: 220614055
2018-11-08 20:04:32 +08:00
|
|
|
// index-type
|
2018-10-07 08:21:53 +08:00
|
|
|
case Token::kw_index:
|
|
|
|
consumeToken(Token::kw_index);
|
|
|
|
return builder.getIndexType();
|
Enable arithmetics for index types.
Arithmetic and comparison instructions are necessary to implement, e.g.,
control flow when lowering MLFunctions to CFGFunctions. (While it is possible
to replace some of the arithmetics by affine_apply instructions for loop
bounds, it is still necessary for loop bounds checking, steps, if-conditions,
non-trivial memref subscripts, etc.) Furthermore, working with indirect
accesses in, e.g., lookup tables for large embeddings, may require operating on
tensors of indexes. For example, the equivalents to C code "LUT[Index[i]]" or
"ResultIndex[i] = i + j" where i, j are loop induction variables require the
arithmetics on indices as well as the possibility to operate on tensors
thereof. Allow arithmetic and comparison operations to apply to index types by
declaring them integer-like. Allow tensors whose element type is index for
indirection purposes.
The absence of vectors with "index" element type is explicitly tested, but the
only justification for this restriction in the CL introducing the test is
"because we don't need them". Do NOT enable vectors of index types, although
it makes vector and tensor types inconsistent with respect to allowed element
types.
PiperOrigin-RevId: 220614055
2018-11-08 20:04:32 +08:00
|
|
|
|
2019-01-08 10:42:04 +08:00
|
|
|
// extended type
|
2019-01-03 06:16:40 +08:00
|
|
|
case Token::exclamation_identifier:
|
2019-01-08 10:42:04 +08:00
|
|
|
return parseExtendedType();
|
2018-06-23 06:52:02 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-02-06 03:47:02 +08:00
|
|
|
/// Parse an arbitrary type.
|
|
|
|
///
|
|
|
|
/// type ::= function-type
|
|
|
|
/// | non-function-type
|
|
|
|
///
|
|
|
|
Type Parser::parseType() {
|
|
|
|
if (getToken().is(Token::l_paren))
|
|
|
|
return parseFunctionType();
|
|
|
|
return parseNonFunctionType();
|
|
|
|
}
|
|
|
|
|
2018-06-23 06:52:02 +08:00
|
|
|
/// Parse a vector type.
|
|
|
|
///
|
2019-02-08 00:36:50 +08:00
|
|
|
/// vector-type ::= `vector` `<` static-dimension-list primitive-type `>`
|
|
|
|
/// static-dimension-list ::= (decimal-literal `x`)+
|
2018-06-23 06:52:02 +08:00
|
|
|
///
|
2018-10-31 05:59:22 +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
|
|
|
|
2019-01-24 06:39:45 +08:00
|
|
|
SmallVector<int64_t, 4> dimensions;
|
2019-02-08 00:36:50 +08:00
|
|
|
if (parseDimensionListRanked(dimensions, /*allowDynamic=*/false))
|
|
|
|
return nullptr;
|
|
|
|
if (dimensions.empty())
|
|
|
|
return (emitError("expected dimension size in vector type"), 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();
|
2018-10-31 05:59:22 +08:00
|
|
|
auto elementType = parseType();
|
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 (!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
|
|
|
|
2018-11-09 05:41:21 +08:00
|
|
|
return VectorType::getChecked(dimensions, elementType,
|
|
|
|
getEncodedSourceLocation(typeLoc));
|
2018-06-23 06:52:02 +08:00
|
|
|
}
|
|
|
|
|
2018-09-14 01:43:35 +08:00
|
|
|
/// Parse an 'x' token in a dimension list, handling the case where the x is
|
|
|
|
/// juxtaposed with an element type, as in "xf32", leaving the "f32" as the next
|
|
|
|
/// token.
|
|
|
|
ParseResult Parser::parseXInDimensionList() {
|
|
|
|
if (getToken().isNot(Token::bare_identifier) || getTokenSpelling()[0] != 'x')
|
|
|
|
return emitError("expected 'x' in dimension list");
|
|
|
|
|
|
|
|
// If we had a prefix of 'x', lex the next token immediately after the 'x'.
|
|
|
|
if (getTokenSpelling().size() != 1)
|
|
|
|
state.lex.resetPointer(getTokenSpelling().data() + 1);
|
|
|
|
|
|
|
|
// Consume the 'x'.
|
|
|
|
consumeToken(Token::bare_identifier);
|
|
|
|
|
|
|
|
return ParseSuccess;
|
|
|
|
}
|
|
|
|
|
2018-06-23 06:52:02 +08:00
|
|
|
/// Parse a dimension list of a tensor or memref type. This populates the
|
2019-02-08 00:36:50 +08:00
|
|
|
/// dimension list, using -1 for the `?` dimensions if `allowDynamic` is set and
|
|
|
|
/// errors out on `?` otherwise.
|
2018-06-23 06:52:02 +08:00
|
|
|
///
|
|
|
|
/// dimension-list-ranked ::= (dimension `x`)*
|
2019-02-08 00:36:50 +08:00
|
|
|
/// dimension ::= `?` | decimal-literal
|
|
|
|
///
|
|
|
|
/// When `allowDynamic` is not set, this can be also used to parse
|
2018-06-23 06:52:02 +08:00
|
|
|
///
|
2019-02-08 00:36:50 +08:00
|
|
|
/// static-dimension-list ::= (decimal-literal `x`)*
|
2019-01-24 06:39:45 +08:00
|
|
|
ParseResult
|
2019-02-08 00:36:50 +08:00
|
|
|
Parser::parseDimensionListRanked(SmallVectorImpl<int64_t> &dimensions,
|
|
|
|
bool allowDynamic = true) {
|
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)) {
|
2019-02-08 00:36:50 +08:00
|
|
|
if (!allowDynamic)
|
|
|
|
return emitError("expected static shape");
|
2018-06-23 06:52:02 +08:00
|
|
|
dimensions.push_back(-1);
|
|
|
|
} else {
|
2019-02-08 00:36:50 +08:00
|
|
|
// Hexadecimal integer literals (starting with `0x`) are not allowed in
|
|
|
|
// aggregate type declarations. Therefore, `0xf32` should be processed as
|
|
|
|
// a sequence of separate elements `0`, `x`, `f32`.
|
|
|
|
if (getTokenSpelling().size() > 1 && getTokenSpelling()[1] == 'x') {
|
|
|
|
// We can get here only if the token is an integer literal. Hexadecimal
|
|
|
|
// integer literals can only start with `0x` (`1x` wouldn't lex as a
|
|
|
|
// literal, just `1` would, at which point we don't get into this
|
|
|
|
// branch).
|
|
|
|
assert(getTokenSpelling()[0] == '0' && "invalid integer literal");
|
|
|
|
dimensions.push_back(0);
|
|
|
|
state.lex.resetPointer(getTokenSpelling().data() + 1);
|
|
|
|
consumeToken();
|
|
|
|
} else {
|
|
|
|
// Make sure this integer value is in bound and valid.
|
|
|
|
auto dimension = getToken().getUnsignedIntegerValue();
|
|
|
|
if (!dimension.hasValue())
|
|
|
|
return emitError("invalid dimension");
|
|
|
|
dimensions.push_back((int64_t)dimension.getValue());
|
|
|
|
consumeToken(Token::integer);
|
|
|
|
}
|
2018-06-23 06:52:02 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
// Make sure we have an 'x' or something like 'xbf32'.
|
2018-09-14 01:43:35 +08:00
|
|
|
if (parseXInDimensionList())
|
|
|
|
return ParseFailure;
|
2018-06-23 06:52:02 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
return ParseSuccess;
|
|
|
|
}
|
|
|
|
|
2019-01-08 10:42:04 +08:00
|
|
|
/// Parse an extended type.
|
2019-01-03 06:16:40 +08:00
|
|
|
///
|
2019-01-08 10:42:04 +08:00
|
|
|
/// extended-type ::= (dialect-type | type-alias)
|
|
|
|
/// dialect-type ::= `!` dialect-namespace `<` '"' type-data '"' `>`
|
|
|
|
/// type-alias ::= `!` alias-name
|
2019-01-03 06:16:40 +08:00
|
|
|
///
|
2019-01-08 10:42:04 +08:00
|
|
|
Type Parser::parseExtendedType() {
|
2019-01-03 06:16:40 +08:00
|
|
|
assert(getToken().is(Token::exclamation_identifier));
|
|
|
|
|
|
|
|
// Parse the dialect namespace.
|
2019-01-08 10:42:04 +08:00
|
|
|
StringRef identifier = getTokenSpelling().drop_front();
|
2019-01-03 06:16:40 +08:00
|
|
|
consumeToken(Token::exclamation_identifier);
|
|
|
|
|
2019-01-08 10:42:04 +08:00
|
|
|
// If there is not a '<' token, we are parsing a type alias.
|
|
|
|
if (getToken().isNot(Token::less)) {
|
|
|
|
// Check for an alias for this type.
|
|
|
|
auto aliasIt = state.typeAliasDefinitions.find(identifier);
|
|
|
|
if (aliasIt == state.typeAliasDefinitions.end())
|
|
|
|
return (emitError("undefined type alias id '" + identifier + "'"),
|
|
|
|
nullptr);
|
|
|
|
return aliasIt->second;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Otherwise, check for a registered dialect with this name.
|
|
|
|
auto *dialect = state.context->getRegisteredDialect(identifier);
|
2019-01-08 01:58:34 +08:00
|
|
|
if (dialect) {
|
|
|
|
// Make sure that the dialect provides a parsing hook.
|
|
|
|
if (!dialect->typeParseHook)
|
|
|
|
return (emitError("dialect '" + dialect->getNamespace() +
|
|
|
|
"' provides no type parsing hook"),
|
|
|
|
nullptr);
|
|
|
|
}
|
2019-01-03 06:16:40 +08:00
|
|
|
|
|
|
|
// Consume the '<'.
|
|
|
|
if (parseToken(Token::less, "expected '<' in dialect type"))
|
|
|
|
return nullptr;
|
|
|
|
|
|
|
|
// Parse the type specific data.
|
|
|
|
if (getToken().isNot(Token::string))
|
|
|
|
return (emitError("expected string literal type data in dialect type"),
|
|
|
|
nullptr);
|
|
|
|
|
|
|
|
auto typeData = getToken().getStringValue();
|
|
|
|
auto loc = getEncodedSourceLocation(getToken().getLoc());
|
|
|
|
consumeToken(Token::string);
|
|
|
|
|
2019-01-08 01:58:34 +08:00
|
|
|
Type result;
|
|
|
|
|
|
|
|
// If we found a registered dialect, then ask it to parse the type.
|
|
|
|
if (dialect) {
|
|
|
|
result = dialect->typeParseHook(typeData, loc, state.context);
|
|
|
|
if (!result)
|
|
|
|
return nullptr;
|
|
|
|
} else {
|
|
|
|
// Otherwise, form a new unknown type.
|
2019-01-08 10:42:04 +08:00
|
|
|
result = UnknownType::get(Identifier::get(identifier, state.context),
|
2019-01-08 01:58:34 +08:00
|
|
|
typeData, state.context);
|
|
|
|
}
|
2019-01-03 06:16:40 +08:00
|
|
|
|
|
|
|
// Consume the '>'.
|
|
|
|
if (parseToken(Token::greater, "expected '>' in dialect type"))
|
|
|
|
return nullptr;
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
2018-06-23 06:52:02 +08:00
|
|
|
/// Parse a tensor type.
|
|
|
|
///
|
|
|
|
/// tensor-type ::= `tensor` `<` dimension-list element-type `>`
|
2018-09-14 01:43:35 +08:00
|
|
|
/// dimension-list ::= dimension-list-ranked | `*x`
|
2018-06-23 06:52:02 +08:00
|
|
|
///
|
2018-10-31 05:59:22 +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;
|
2019-01-24 06:39:45 +08:00
|
|
|
SmallVector<int64_t, 4> dimensions;
|
2018-06-23 06:52:02 +08:00
|
|
|
|
2018-09-14 01:43:35 +08:00
|
|
|
if (consumeIf(Token::star)) {
|
|
|
|
// This is an unranked tensor type.
|
2018-06-23 06:52:02 +08:00
|
|
|
isUnranked = true;
|
2018-09-14 01:43:35 +08:00
|
|
|
|
|
|
|
if (parseXInDimensionList())
|
|
|
|
return nullptr;
|
|
|
|
|
2018-06-23 06:52:02 +08:00
|
|
|
} 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.
|
2018-11-09 05:47:19 +08:00
|
|
|
auto typeLocation = getEncodedSourceLocation(getToken().getLoc());
|
2018-10-31 05:59:22 +08:00
|
|
|
auto elementType = parseType();
|
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 (!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
|
|
|
|
2018-06-24 09:09:09 +08:00
|
|
|
if (isUnranked)
|
2018-11-09 05:47:19 +08:00
|
|
|
return UnrankedTensorType::getChecked(elementType, typeLocation);
|
|
|
|
return RankedTensorType::getChecked(dimensions, elementType, typeLocation);
|
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-10-31 05:59:22 +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
|
|
|
|
2019-01-24 06:39:45 +08:00
|
|
|
SmallVector<int64_t, 4> dimensions;
|
2018-06-23 06:52:02 +08:00
|
|
|
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();
|
2018-10-31 05:59:22 +08:00
|
|
|
auto elementType = parseType();
|
2018-06-23 13:03:48 +08:00
|
|
|
if (!elementType)
|
|
|
|
return nullptr;
|
2018-06-23 06:52:02 +08:00
|
|
|
|
2018-07-17 00:45:22 +08:00
|
|
|
// Parse semi-affine-map-composition.
|
2018-10-10 07:39:24 +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-10-10 07:39:24 +08:00
|
|
|
auto affineMap = parseAffineMapReference();
|
|
|
|
if (!affineMap)
|
2018-07-17 00:45:22 +08:00
|
|
|
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
|
|
|
}
|
|
|
|
|
2018-11-02 16:48:22 +08:00
|
|
|
return MemRefType::getChecked(dimensions, elementType, affineMapComposition,
|
|
|
|
memorySpace, getEncodedSourceLocation(typeLoc));
|
2018-06-23 06:52:02 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
/// Parse a function type.
|
|
|
|
///
|
|
|
|
/// function-type ::= type-list-parens `->` type-list
|
|
|
|
///
|
2018-10-31 05:59:22 +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-10-31 05:59:22 +08:00
|
|
|
SmallVector<Type, 4> arguments, results;
|
2019-02-06 03:47:02 +08:00
|
|
|
if (parseTypeListParens(arguments) ||
|
2018-07-24 08:30:01 +08:00
|
|
|
parseToken(Token::arrow, "expected '->' in function type") ||
|
2019-02-06 03:47:02 +08:00
|
|
|
parseFunctionResultTypes(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)*
|
|
|
|
///
|
2018-10-31 05:59:22 +08:00
|
|
|
ParseResult Parser::parseTypeListNoParens(SmallVectorImpl<Type> &elements) {
|
2018-07-23 23:42:19 +08:00
|
|
|
auto parseElt = [&]() -> ParseResult {
|
|
|
|
auto elt = parseType();
|
|
|
|
elements.push_back(elt);
|
|
|
|
return elt ? ParseSuccess : ParseFailure;
|
|
|
|
};
|
|
|
|
|
|
|
|
return parseCommaSeparatedList(parseElt);
|
|
|
|
}
|
|
|
|
|
2019-02-06 03:47:02 +08:00
|
|
|
/// Parse a parenthesized list of types.
|
2018-06-23 06:52:02 +08:00
|
|
|
///
|
|
|
|
/// type-list-parens ::= `(` `)`
|
2018-07-23 23:42:19 +08:00
|
|
|
/// | `(` type-list-no-parens `)`
|
2018-06-23 06:52:02 +08:00
|
|
|
///
|
2019-02-06 03:47:02 +08:00
|
|
|
ParseResult Parser::parseTypeListParens(SmallVectorImpl<Type> &elements) {
|
|
|
|
if (parseToken(Token::l_paren, "expected '('"))
|
|
|
|
return ParseFailure;
|
2018-06-23 13:03:48 +08:00
|
|
|
|
2019-02-06 03:47:02 +08:00
|
|
|
// Handle empty lists.
|
|
|
|
if (getToken().is(Token::r_paren))
|
|
|
|
return consumeToken(), ParseSuccess;
|
2018-06-23 06:52:02 +08:00
|
|
|
|
2019-02-06 03:47:02 +08:00
|
|
|
if (parseTypeListNoParens(elements) ||
|
|
|
|
parseToken(Token::r_paren, "expected ')'"))
|
2018-06-23 06:52:02 +08:00
|
|
|
return ParseFailure;
|
2019-02-06 03:47:02 +08:00
|
|
|
return ParseSuccess;
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Parse a function result type.
|
|
|
|
///
|
|
|
|
/// function-result-type ::= type-list-parens
|
|
|
|
/// | non-function-type
|
|
|
|
///
|
|
|
|
ParseResult Parser::parseFunctionResultTypes(SmallVectorImpl<Type> &elements) {
|
|
|
|
if (getToken().is(Token::l_paren))
|
|
|
|
return parseTypeListParens(elements);
|
2018-06-23 06:52:02 +08:00
|
|
|
|
2019-02-06 03:47:02 +08:00
|
|
|
Type t = parseNonFunctionType();
|
|
|
|
if (!t)
|
|
|
|
return ParseFailure;
|
|
|
|
elements.push_back(t);
|
2018-06-23 06:52:02 +08:00
|
|
|
return ParseSuccess;
|
|
|
|
}
|
|
|
|
|
2018-07-05 11:45:39 +08:00
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
// Attribute parsing.
|
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
|
2018-10-19 04:54:44 +08:00
|
|
|
namespace {
|
|
|
|
class TensorLiteralParser {
|
|
|
|
public:
|
2019-01-18 06:11:05 +08:00
|
|
|
TensorLiteralParser(Parser &p, Type eltTy) : p(p), eltTy(eltTy) {}
|
2018-10-19 04:54:44 +08:00
|
|
|
|
2019-01-12 03:23:15 +08:00
|
|
|
ParseResult parse() {
|
|
|
|
if (p.getToken().isNot(Token::l_square))
|
|
|
|
return p.emitError("expected '[' in tensor literal list");
|
|
|
|
return parseList(shape);
|
|
|
|
}
|
2018-10-19 04:54:44 +08:00
|
|
|
|
2019-01-18 06:11:05 +08:00
|
|
|
ArrayRef<Attribute> getValues() const { return storage; }
|
2018-10-19 04:54:44 +08:00
|
|
|
|
2019-01-24 06:39:45 +08:00
|
|
|
ArrayRef<int64_t> getShape() const { return shape; }
|
2018-10-19 04:54:44 +08:00
|
|
|
|
|
|
|
private:
|
|
|
|
/// Parse either a single element or a list of elements. Return the dimensions
|
|
|
|
/// of the parsed sub-tensor in dims.
|
2019-01-24 06:39:45 +08:00
|
|
|
ParseResult parseElementOrList(llvm::SmallVectorImpl<int64_t> &dims);
|
2018-10-19 04:54:44 +08:00
|
|
|
|
|
|
|
/// Parse a list of either lists or elements, returning the dimensions of the
|
|
|
|
/// parsed sub-tensors in dims. For example:
|
|
|
|
/// parseList([1, 2, 3]) -> Success, [3]
|
|
|
|
/// parseList([[1, 2], [3, 4]]) -> Success, [2, 2]
|
|
|
|
/// parseList([[1, 2], 3]) -> Failure
|
|
|
|
/// parseList([[1, [2, 3]], [4, [5]]]) -> Failure
|
2019-01-24 06:39:45 +08:00
|
|
|
ParseResult parseList(llvm::SmallVectorImpl<int64_t> &dims);
|
2018-10-19 04:54:44 +08:00
|
|
|
|
|
|
|
Parser &p;
|
2018-10-31 05:59:22 +08:00
|
|
|
Type eltTy;
|
2019-01-24 06:39:45 +08:00
|
|
|
SmallVector<int64_t, 4> shape;
|
2019-01-18 06:11:05 +08:00
|
|
|
std::vector<Attribute> storage;
|
2018-10-19 04:54:44 +08:00
|
|
|
};
|
|
|
|
} // namespace
|
|
|
|
|
|
|
|
/// Parse either a single element or a list of elements. Return the dimensions
|
|
|
|
/// of the parsed sub-tensor in dims.
|
|
|
|
ParseResult
|
2019-01-24 06:39:45 +08:00
|
|
|
TensorLiteralParser::parseElementOrList(llvm::SmallVectorImpl<int64_t> &dims) {
|
2018-10-19 04:54:44 +08:00
|
|
|
switch (p.getToken().getKind()) {
|
|
|
|
case Token::l_square:
|
|
|
|
return parseList(dims);
|
|
|
|
case Token::floatliteral:
|
|
|
|
case Token::integer:
|
|
|
|
case Token::minus: {
|
2018-12-17 23:19:53 +08:00
|
|
|
auto result = p.parseAttribute(eltTy);
|
2018-10-19 04:54:44 +08:00
|
|
|
if (!result)
|
2018-12-17 23:19:53 +08:00
|
|
|
return ParseResult::ParseFailure;
|
2018-10-19 04:54:44 +08:00
|
|
|
// check result matches the element type.
|
2018-10-31 05:59:22 +08:00
|
|
|
switch (eltTy.getKind()) {
|
2019-01-04 06:29:52 +08:00
|
|
|
case StandardTypes::BF16:
|
|
|
|
case StandardTypes::F16:
|
|
|
|
case StandardTypes::F32:
|
|
|
|
case StandardTypes::F64: {
|
2018-12-18 21:25:17 +08:00
|
|
|
// Bitcast the APFloat value to APInt and store the bit representation.
|
|
|
|
auto fpAttrResult = result.dyn_cast<FloatAttr>();
|
|
|
|
if (!fpAttrResult)
|
2018-12-17 23:19:53 +08:00
|
|
|
return p.emitError(
|
2018-12-18 21:25:17 +08:00
|
|
|
"expected tensor literal element with floating point type");
|
|
|
|
auto apInt = fpAttrResult.getValue().bitcastToAPInt();
|
|
|
|
|
|
|
|
// FIXME: using 64 bits and double semantics for BF16 because APFloat does
|
|
|
|
// not support BF16 directly.
|
|
|
|
size_t bitWidth = eltTy.isBF16() ? 64 : eltTy.getIntOrFloatBitWidth();
|
|
|
|
assert(apInt.getBitWidth() == bitWidth);
|
|
|
|
(void)bitWidth;
|
2019-01-18 06:11:05 +08:00
|
|
|
(void)apInt;
|
2018-10-19 04:54:44 +08:00
|
|
|
break;
|
|
|
|
}
|
2019-01-04 06:29:52 +08:00
|
|
|
case StandardTypes::Integer: {
|
2018-10-26 06:46:10 +08:00
|
|
|
if (!result.isa<IntegerAttr>())
|
2018-10-19 04:54:44 +08:00
|
|
|
return p.emitError("expected tensor literal element has integer type");
|
2018-10-26 06:46:10 +08:00
|
|
|
auto value = result.cast<IntegerAttr>().getValue();
|
2019-01-18 06:11:05 +08:00
|
|
|
if (value.getMinSignedBits() > eltTy.getIntOrFloatBitWidth())
|
2018-10-19 04:54:44 +08:00
|
|
|
return p.emitError("tensor literal element has more bits than that "
|
|
|
|
"specified in the type");
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
default:
|
|
|
|
return p.emitError("expected integer or float tensor element");
|
|
|
|
}
|
2019-01-18 06:11:05 +08:00
|
|
|
storage.push_back(result);
|
2018-10-19 04:54:44 +08:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
default:
|
|
|
|
return p.emitError("expected '[' or scalar constant inside tensor literal");
|
|
|
|
}
|
|
|
|
return ParseSuccess;
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Parse a list of either lists or elements, returning the dimensions of the
|
|
|
|
/// parsed sub-tensors in dims. For example:
|
|
|
|
/// parseList([1, 2, 3]) -> Success, [3]
|
|
|
|
/// parseList([[1, 2], [3, 4]]) -> Success, [2, 2]
|
|
|
|
/// parseList([[1, 2], 3]) -> Failure
|
|
|
|
/// parseList([[1, [2, 3]], [4, [5]]]) -> Failure
|
2019-01-24 06:39:45 +08:00
|
|
|
ParseResult
|
|
|
|
TensorLiteralParser::parseList(llvm::SmallVectorImpl<int64_t> &dims) {
|
2018-10-19 04:54:44 +08:00
|
|
|
p.consumeToken(Token::l_square);
|
|
|
|
|
2019-01-24 06:39:45 +08:00
|
|
|
auto checkDims = [&](const llvm::SmallVectorImpl<int64_t> &prevDims,
|
|
|
|
const llvm::SmallVectorImpl<int64_t> &newDims) {
|
2018-10-19 04:54:44 +08:00
|
|
|
if (prevDims == newDims)
|
|
|
|
return ParseSuccess;
|
|
|
|
return p.emitError("tensor literal is invalid; ranks are not consistent "
|
|
|
|
"between elements");
|
|
|
|
};
|
|
|
|
|
|
|
|
bool first = true;
|
2019-01-24 06:39:45 +08:00
|
|
|
llvm::SmallVector<int64_t, 4> newDims;
|
2018-10-19 04:54:44 +08:00
|
|
|
unsigned size = 0;
|
|
|
|
auto parseCommaSeparatedList = [&]() {
|
2019-01-24 06:39:45 +08:00
|
|
|
llvm::SmallVector<int64_t, 4> thisDims;
|
2018-10-19 04:54:44 +08:00
|
|
|
if (parseElementOrList(thisDims))
|
|
|
|
return ParseFailure;
|
|
|
|
++size;
|
|
|
|
if (!first)
|
|
|
|
return checkDims(newDims, thisDims);
|
|
|
|
newDims = thisDims;
|
|
|
|
first = false;
|
|
|
|
return ParseSuccess;
|
|
|
|
};
|
|
|
|
if (p.parseCommaSeparatedListUntil(Token::r_square, parseCommaSeparatedList))
|
|
|
|
return ParseFailure;
|
|
|
|
|
|
|
|
// Return the sublists' dimensions with 'size' prepended.
|
|
|
|
dims.clear();
|
|
|
|
dims.push_back(size);
|
2019-01-08 07:06:32 +08:00
|
|
|
dims.append(newDims.begin(), newDims.end());
|
2018-10-19 04:54:44 +08:00
|
|
|
return ParseSuccess;
|
|
|
|
}
|
|
|
|
|
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,
|
2018-10-31 05:59:22 +08:00
|
|
|
FunctionType type) {
|
2018-08-22 08:55:22 +08:00
|
|
|
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];
|
2018-09-08 00:08:13 +08:00
|
|
|
if (!entry)
|
2019-01-03 02:20:00 +08:00
|
|
|
entry = new Function(getEncodedSourceLocation(nameLoc), name, type,
|
2018-12-28 03:07:34 +08:00
|
|
|
/*attrs=*/{});
|
2018-09-08 00:08:13 +08:00
|
|
|
function = entry;
|
2018-08-22 08:55:22 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
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
|
2018-11-16 09:53:51 +08:00
|
|
|
/// | integer-literal (`:` integer-type)
|
|
|
|
/// | float-literal (`:` float-type)
|
2018-07-05 11:45:39 +08:00
|
|
|
/// | 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-10-19 04:54:44 +08:00
|
|
|
/// | (`splat<` | `dense<`) (tensor-type | vector-type)`,`
|
2018-10-10 23:57:51 +08:00
|
|
|
/// attribute-value `>`
|
Add support to constant sparse tensor / vector attribute
The SparseElementsAttr uses (COO) Coordinate List encoding to represents a
sparse tensor / vector. Specifically, the coordinates and values are stored as
two dense elements attributes. The first dense elements attribute is a 2-D
attribute with shape [N, ndims], which contains the indices of the elements
with nonzero values in the constant vector/tensor. The second elements
attribute is a 1-D attribute list with shape [N], which supplies the values for
each element in the first elements attribute. ndims is the rank of the
vector/tensor and N is the total nonzero elements.
The syntax is:
`sparse<` (tensor-type | vector-type)`, ` indices-attribute-list, values-attribute-list `>`
Example: a sparse tensor
sparse<vector<3x4xi32>, [[0, 0], [1, 2]], [1, 2]> represents the dense tensor
[[1, 0, 0, 0]
[0, 0, 2, 0]
[0, 0, 0, 0]]
PiperOrigin-RevId: 217764319
2018-10-19 05:02:20 +08:00
|
|
|
/// | `sparse<` (tensor-type | vector-type)`,`
|
|
|
|
/// attribute-value`, ` attribute-value `>`
|
2018-07-05 11:45:39 +08:00
|
|
|
///
|
2018-11-16 09:53:51 +08:00
|
|
|
Attribute Parser::parseAttribute(Type type) {
|
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);
|
2019-01-14 21:37:14 +08:00
|
|
|
auto valTok = getToken().getLoc();
|
2018-08-01 08:15:15 +08:00
|
|
|
consumeToken(Token::floatliteral);
|
2018-11-16 09:53:51 +08:00
|
|
|
if (!type) {
|
|
|
|
if (consumeIf(Token::colon)) {
|
|
|
|
if (!(type = parseType()))
|
|
|
|
return nullptr;
|
|
|
|
} else {
|
2018-12-17 23:19:53 +08:00
|
|
|
// Default to F64 when no type is specified.
|
|
|
|
type = builder.getF64Type();
|
2018-11-16 09:53:51 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
if (!type.isa<FloatType>())
|
|
|
|
return (emitError("floating point value not valid for specified type"),
|
|
|
|
nullptr);
|
2019-01-14 21:37:14 +08:00
|
|
|
return FloatAttr::getChecked(type, val.getValue(),
|
|
|
|
getEncodedSourceLocation(valTok));
|
2018-08-01 08:15:15 +08:00
|
|
|
}
|
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)
|
2018-12-17 23:19:53 +08:00
|
|
|
return (emitError("integer constant out of range for attribute"),
|
|
|
|
nullptr);
|
2018-07-05 11:45:39 +08:00
|
|
|
consumeToken(Token::integer);
|
2018-11-16 09:53:51 +08:00
|
|
|
if (!type) {
|
|
|
|
if (consumeIf(Token::colon)) {
|
|
|
|
if (!(type = parseType()))
|
|
|
|
return nullptr;
|
|
|
|
} else {
|
|
|
|
// Default to i64 if not type is specified.
|
|
|
|
type = builder.getIntegerType(64);
|
|
|
|
}
|
|
|
|
}
|
2018-12-05 20:31:59 +08:00
|
|
|
if (!type.isIntOrIndex())
|
2018-11-16 09:53:51 +08:00
|
|
|
return (emitError("integer value not valid for specified type"), nullptr);
|
2018-12-18 02:05:56 +08:00
|
|
|
int width = type.isIndex() ? 64 : type.getIntOrFloatBitWidth();
|
2018-12-17 23:19:53 +08:00
|
|
|
APInt apInt(width, val.getValue());
|
|
|
|
if (apInt != *val)
|
|
|
|
return emitError("integer constant out of range for attribute"), nullptr;
|
|
|
|
return builder.getIntegerAttr(type, apInt);
|
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)
|
2018-12-17 23:19:53 +08:00
|
|
|
return (emitError("integer constant out of range for attribute"),
|
|
|
|
nullptr);
|
2018-07-05 11:45:39 +08:00
|
|
|
consumeToken(Token::integer);
|
2018-11-16 09:53:51 +08:00
|
|
|
if (!type) {
|
|
|
|
if (consumeIf(Token::colon)) {
|
|
|
|
if (!(type = parseType()))
|
|
|
|
return nullptr;
|
|
|
|
} else {
|
|
|
|
// Default to i64 if not type is specified.
|
|
|
|
type = builder.getIntegerType(64);
|
|
|
|
}
|
|
|
|
}
|
2018-12-05 20:31:59 +08:00
|
|
|
if (!type.isIntOrIndex())
|
2018-11-16 09:53:51 +08:00
|
|
|
return (emitError("integer value not valid for type"), nullptr);
|
2018-12-18 02:05:56 +08:00
|
|
|
int width = type.isIndex() ? 64 : type.getIntOrFloatBitWidth();
|
2018-12-17 23:19:53 +08:00
|
|
|
APInt apInt(width, *val, /*isSigned=*/true);
|
|
|
|
if (apInt != *val)
|
|
|
|
return (emitError("integer constant out of range for attribute"),
|
|
|
|
nullptr);
|
|
|
|
return builder.getIntegerAttr(type, -apInt);
|
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);
|
2019-01-14 21:37:14 +08:00
|
|
|
auto valTok = getToken().getLoc();
|
2018-08-01 08:15:15 +08:00
|
|
|
consumeToken(Token::floatliteral);
|
2018-11-16 09:53:51 +08:00
|
|
|
if (!type) {
|
|
|
|
if (consumeIf(Token::colon)) {
|
|
|
|
if (!(type = parseType()))
|
|
|
|
return nullptr;
|
|
|
|
} else {
|
2018-12-28 08:51:09 +08:00
|
|
|
// Default to F64 when no type is specified.
|
|
|
|
type = builder.getF64Type();
|
2018-11-16 09:53:51 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
if (!type.isa<FloatType>())
|
|
|
|
return (emitError("floating point value not valid for type"), nullptr);
|
2019-01-14 21:37:14 +08:00
|
|
|
return FloatAttr::getChecked(type, -val.getValue(),
|
|
|
|
getEncodedSourceLocation(valTok));
|
2018-08-01 08:15:15 +08:00
|
|
|
}
|
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-10-26 06:46:10 +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-10-26 13:13:03 +08:00
|
|
|
// Try to parse an affine map or an integer set reference.
|
|
|
|
AffineMap map;
|
|
|
|
IntegerSet set;
|
2019-01-27 02:41:17 +08:00
|
|
|
if (parseAffineMapOrIntegerSetReference(map, set))
|
2019-01-29 13:23:53 +08:00
|
|
|
return nullptr;
|
2018-10-26 13:13:03 +08:00
|
|
|
if (map)
|
|
|
|
return builder.getAffineMapAttr(map);
|
2019-01-27 02:41:17 +08:00
|
|
|
assert(set);
|
|
|
|
return builder.getIntegerSetAttr(set);
|
2018-07-05 11:45:39 +08:00
|
|
|
}
|
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();
|
2018-10-31 05:59:22 +08:00
|
|
|
Type type = parseType();
|
2018-08-20 12:17:22 +08:00
|
|
|
if (!type)
|
|
|
|
return nullptr;
|
2018-10-31 05:59:22 +08:00
|
|
|
auto fnType = type.dyn_cast<FunctionType>();
|
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-10-24 04:44:04 +08:00
|
|
|
case Token::kw_opaque: {
|
|
|
|
consumeToken(Token::kw_opaque);
|
|
|
|
if (parseToken(Token::less, "expected '<' after 'opaque'"))
|
|
|
|
return nullptr;
|
2018-10-31 05:59:22 +08:00
|
|
|
auto type = parseVectorOrTensorType();
|
2018-10-24 04:44:04 +08:00
|
|
|
if (!type)
|
|
|
|
return nullptr;
|
2019-01-12 01:03:34 +08:00
|
|
|
if (getToken().getKind() != Token::string)
|
|
|
|
return (emitError("opaque string should start with '0x'"), nullptr);
|
2018-10-24 04:44:04 +08:00
|
|
|
auto val = getToken().getStringValue();
|
|
|
|
if (val.size() < 2 || val[0] != '0' || val[1] != 'x')
|
|
|
|
return (emitError("opaque string should start with '0x'"), nullptr);
|
|
|
|
val = val.substr(2);
|
|
|
|
if (!std::all_of(val.begin(), val.end(),
|
|
|
|
[](char c) { return llvm::isHexDigit(c); })) {
|
|
|
|
return (emitError("opaque string only contains hex digits"), nullptr);
|
|
|
|
}
|
|
|
|
consumeToken(Token::string);
|
|
|
|
if (parseToken(Token::greater, "expected '>'"))
|
|
|
|
return nullptr;
|
|
|
|
return builder.getOpaqueElementsAttr(type, llvm::fromHex(val));
|
|
|
|
}
|
2018-10-10 23:57:51 +08:00
|
|
|
case Token::kw_splat: {
|
|
|
|
consumeToken(Token::kw_splat);
|
2018-10-19 04:54:44 +08:00
|
|
|
if (parseToken(Token::less, "expected '<' after 'splat'"))
|
2018-10-10 23:57:51 +08:00
|
|
|
return nullptr;
|
|
|
|
|
2018-10-31 05:59:22 +08:00
|
|
|
auto type = parseVectorOrTensorType();
|
2018-10-19 04:54:44 +08:00
|
|
|
if (!type)
|
2018-10-10 23:57:51 +08:00
|
|
|
return nullptr;
|
|
|
|
switch (getToken().getKind()) {
|
|
|
|
case Token::floatliteral:
|
|
|
|
case Token::integer:
|
|
|
|
case Token::minus: {
|
2018-11-16 09:53:51 +08:00
|
|
|
auto scalar = parseAttribute(type.getElementType());
|
2019-01-05 03:27:58 +08:00
|
|
|
if (!scalar)
|
|
|
|
return nullptr;
|
2018-10-10 23:57:51 +08:00
|
|
|
if (parseToken(Token::greater, "expected '>'"))
|
|
|
|
return nullptr;
|
|
|
|
return builder.getSplatElementsAttr(type, scalar);
|
|
|
|
}
|
|
|
|
default:
|
2018-10-19 04:54:44 +08:00
|
|
|
return (emitError("expected scalar constant inside tensor literal"),
|
|
|
|
nullptr);
|
2018-10-10 23:57:51 +08:00
|
|
|
}
|
|
|
|
}
|
2018-10-19 04:54:44 +08:00
|
|
|
case Token::kw_dense: {
|
|
|
|
consumeToken(Token::kw_dense);
|
|
|
|
if (parseToken(Token::less, "expected '<' after 'dense'"))
|
|
|
|
return nullptr;
|
2018-10-10 23:57:51 +08:00
|
|
|
|
2018-10-31 05:59:22 +08:00
|
|
|
auto type = parseVectorOrTensorType();
|
2018-10-19 04:54:44 +08:00
|
|
|
if (!type)
|
|
|
|
return nullptr;
|
|
|
|
|
|
|
|
switch (getToken().getKind()) {
|
|
|
|
case Token::l_square: {
|
|
|
|
auto attr = parseDenseElementsAttr(type);
|
|
|
|
if (!attr)
|
|
|
|
return nullptr;
|
|
|
|
if (parseToken(Token::greater, "expected '>'"))
|
|
|
|
return nullptr;
|
|
|
|
return attr;
|
|
|
|
}
|
|
|
|
default:
|
|
|
|
return (emitError("expected '[' to start dense tensor literal"), nullptr);
|
|
|
|
}
|
|
|
|
}
|
Add support to constant sparse tensor / vector attribute
The SparseElementsAttr uses (COO) Coordinate List encoding to represents a
sparse tensor / vector. Specifically, the coordinates and values are stored as
two dense elements attributes. The first dense elements attribute is a 2-D
attribute with shape [N, ndims], which contains the indices of the elements
with nonzero values in the constant vector/tensor. The second elements
attribute is a 1-D attribute list with shape [N], which supplies the values for
each element in the first elements attribute. ndims is the rank of the
vector/tensor and N is the total nonzero elements.
The syntax is:
`sparse<` (tensor-type | vector-type)`, ` indices-attribute-list, values-attribute-list `>`
Example: a sparse tensor
sparse<vector<3x4xi32>, [[0, 0], [1, 2]], [1, 2]> represents the dense tensor
[[1, 0, 0, 0]
[0, 0, 2, 0]
[0, 0, 0, 0]]
PiperOrigin-RevId: 217764319
2018-10-19 05:02:20 +08:00
|
|
|
case Token::kw_sparse: {
|
|
|
|
consumeToken(Token::kw_sparse);
|
|
|
|
if (parseToken(Token::less, "Expected '<' after 'sparse'"))
|
|
|
|
return nullptr;
|
|
|
|
|
2018-10-31 05:59:22 +08:00
|
|
|
auto type = parseVectorOrTensorType();
|
Add support to constant sparse tensor / vector attribute
The SparseElementsAttr uses (COO) Coordinate List encoding to represents a
sparse tensor / vector. Specifically, the coordinates and values are stored as
two dense elements attributes. The first dense elements attribute is a 2-D
attribute with shape [N, ndims], which contains the indices of the elements
with nonzero values in the constant vector/tensor. The second elements
attribute is a 1-D attribute list with shape [N], which supplies the values for
each element in the first elements attribute. ndims is the rank of the
vector/tensor and N is the total nonzero elements.
The syntax is:
`sparse<` (tensor-type | vector-type)`, ` indices-attribute-list, values-attribute-list `>`
Example: a sparse tensor
sparse<vector<3x4xi32>, [[0, 0], [1, 2]], [1, 2]> represents the dense tensor
[[1, 0, 0, 0]
[0, 0, 2, 0]
[0, 0, 0, 0]]
PiperOrigin-RevId: 217764319
2018-10-19 05:02:20 +08:00
|
|
|
if (!type)
|
|
|
|
return nullptr;
|
|
|
|
|
|
|
|
switch (getToken().getKind()) {
|
|
|
|
case Token::l_square: {
|
|
|
|
/// Parse indices
|
2019-01-20 12:54:09 +08:00
|
|
|
auto indicesEltType = builder.getIntegerType(64);
|
2019-02-08 02:13:50 +08:00
|
|
|
auto indices = parseDenseElementsAttrAsTensor(indicesEltType);
|
2019-01-15 04:23:38 +08:00
|
|
|
if (!indices)
|
|
|
|
return nullptr;
|
Add support to constant sparse tensor / vector attribute
The SparseElementsAttr uses (COO) Coordinate List encoding to represents a
sparse tensor / vector. Specifically, the coordinates and values are stored as
two dense elements attributes. The first dense elements attribute is a 2-D
attribute with shape [N, ndims], which contains the indices of the elements
with nonzero values in the constant vector/tensor. The second elements
attribute is a 1-D attribute list with shape [N], which supplies the values for
each element in the first elements attribute. ndims is the rank of the
vector/tensor and N is the total nonzero elements.
The syntax is:
`sparse<` (tensor-type | vector-type)`, ` indices-attribute-list, values-attribute-list `>`
Example: a sparse tensor
sparse<vector<3x4xi32>, [[0, 0], [1, 2]], [1, 2]> represents the dense tensor
[[1, 0, 0, 0]
[0, 0, 2, 0]
[0, 0, 0, 0]]
PiperOrigin-RevId: 217764319
2018-10-19 05:02:20 +08:00
|
|
|
|
|
|
|
if (parseToken(Token::comma, "expected ','"))
|
|
|
|
return nullptr;
|
|
|
|
|
|
|
|
/// Parse values.
|
2018-10-31 05:59:22 +08:00
|
|
|
auto valuesEltType = type.getElementType();
|
2019-02-08 02:13:50 +08:00
|
|
|
auto values = parseDenseElementsAttrAsTensor(valuesEltType);
|
2019-01-12 02:30:40 +08:00
|
|
|
if (!values)
|
|
|
|
return nullptr;
|
Add support to constant sparse tensor / vector attribute
The SparseElementsAttr uses (COO) Coordinate List encoding to represents a
sparse tensor / vector. Specifically, the coordinates and values are stored as
two dense elements attributes. The first dense elements attribute is a 2-D
attribute with shape [N, ndims], which contains the indices of the elements
with nonzero values in the constant vector/tensor. The second elements
attribute is a 1-D attribute list with shape [N], which supplies the values for
each element in the first elements attribute. ndims is the rank of the
vector/tensor and N is the total nonzero elements.
The syntax is:
`sparse<` (tensor-type | vector-type)`, ` indices-attribute-list, values-attribute-list `>`
Example: a sparse tensor
sparse<vector<3x4xi32>, [[0, 0], [1, 2]], [1, 2]> represents the dense tensor
[[1, 0, 0, 0]
[0, 0, 2, 0]
[0, 0, 0, 0]]
PiperOrigin-RevId: 217764319
2018-10-19 05:02:20 +08:00
|
|
|
|
|
|
|
/// Sanity check.
|
2018-10-31 05:59:22 +08:00
|
|
|
auto indicesType = indices.getType();
|
|
|
|
auto valuesType = values.getType();
|
|
|
|
auto sameShape = (indicesType.getRank() == 1) ||
|
|
|
|
(type.getRank() == indicesType.getDimSize(1));
|
Add support to constant sparse tensor / vector attribute
The SparseElementsAttr uses (COO) Coordinate List encoding to represents a
sparse tensor / vector. Specifically, the coordinates and values are stored as
two dense elements attributes. The first dense elements attribute is a 2-D
attribute with shape [N, ndims], which contains the indices of the elements
with nonzero values in the constant vector/tensor. The second elements
attribute is a 1-D attribute list with shape [N], which supplies the values for
each element in the first elements attribute. ndims is the rank of the
vector/tensor and N is the total nonzero elements.
The syntax is:
`sparse<` (tensor-type | vector-type)`, ` indices-attribute-list, values-attribute-list `>`
Example: a sparse tensor
sparse<vector<3x4xi32>, [[0, 0], [1, 2]], [1, 2]> represents the dense tensor
[[1, 0, 0, 0]
[0, 0, 2, 0]
[0, 0, 0, 0]]
PiperOrigin-RevId: 217764319
2018-10-19 05:02:20 +08:00
|
|
|
auto sameElementNum =
|
2018-10-31 05:59:22 +08:00
|
|
|
indicesType.getDimSize(0) == valuesType.getDimSize(0);
|
Add support to constant sparse tensor / vector attribute
The SparseElementsAttr uses (COO) Coordinate List encoding to represents a
sparse tensor / vector. Specifically, the coordinates and values are stored as
two dense elements attributes. The first dense elements attribute is a 2-D
attribute with shape [N, ndims], which contains the indices of the elements
with nonzero values in the constant vector/tensor. The second elements
attribute is a 1-D attribute list with shape [N], which supplies the values for
each element in the first elements attribute. ndims is the rank of the
vector/tensor and N is the total nonzero elements.
The syntax is:
`sparse<` (tensor-type | vector-type)`, ` indices-attribute-list, values-attribute-list `>`
Example: a sparse tensor
sparse<vector<3x4xi32>, [[0, 0], [1, 2]], [1, 2]> represents the dense tensor
[[1, 0, 0, 0]
[0, 0, 2, 0]
[0, 0, 0, 0]]
PiperOrigin-RevId: 217764319
2018-10-19 05:02:20 +08:00
|
|
|
if (!sameShape || !sameElementNum) {
|
|
|
|
std::string str;
|
|
|
|
llvm::raw_string_ostream s(str);
|
|
|
|
s << "expected shape ([";
|
2018-10-31 05:59:22 +08:00
|
|
|
interleaveComma(type.getShape(), s);
|
Add support to constant sparse tensor / vector attribute
The SparseElementsAttr uses (COO) Coordinate List encoding to represents a
sparse tensor / vector. Specifically, the coordinates and values are stored as
two dense elements attributes. The first dense elements attribute is a 2-D
attribute with shape [N, ndims], which contains the indices of the elements
with nonzero values in the constant vector/tensor. The second elements
attribute is a 1-D attribute list with shape [N], which supplies the values for
each element in the first elements attribute. ndims is the rank of the
vector/tensor and N is the total nonzero elements.
The syntax is:
`sparse<` (tensor-type | vector-type)`, ` indices-attribute-list, values-attribute-list `>`
Example: a sparse tensor
sparse<vector<3x4xi32>, [[0, 0], [1, 2]], [1, 2]> represents the dense tensor
[[1, 0, 0, 0]
[0, 0, 2, 0]
[0, 0, 0, 0]]
PiperOrigin-RevId: 217764319
2018-10-19 05:02:20 +08:00
|
|
|
s << "]); inferred shape of indices literal ([";
|
2018-10-31 05:59:22 +08:00
|
|
|
interleaveComma(indicesType.getShape(), s);
|
Add support to constant sparse tensor / vector attribute
The SparseElementsAttr uses (COO) Coordinate List encoding to represents a
sparse tensor / vector. Specifically, the coordinates and values are stored as
two dense elements attributes. The first dense elements attribute is a 2-D
attribute with shape [N, ndims], which contains the indices of the elements
with nonzero values in the constant vector/tensor. The second elements
attribute is a 1-D attribute list with shape [N], which supplies the values for
each element in the first elements attribute. ndims is the rank of the
vector/tensor and N is the total nonzero elements.
The syntax is:
`sparse<` (tensor-type | vector-type)`, ` indices-attribute-list, values-attribute-list `>`
Example: a sparse tensor
sparse<vector<3x4xi32>, [[0, 0], [1, 2]], [1, 2]> represents the dense tensor
[[1, 0, 0, 0]
[0, 0, 2, 0]
[0, 0, 0, 0]]
PiperOrigin-RevId: 217764319
2018-10-19 05:02:20 +08:00
|
|
|
s << "]); inferred shape of values literal ([";
|
2018-10-31 05:59:22 +08:00
|
|
|
interleaveComma(valuesType.getShape(), s);
|
Add support to constant sparse tensor / vector attribute
The SparseElementsAttr uses (COO) Coordinate List encoding to represents a
sparse tensor / vector. Specifically, the coordinates and values are stored as
two dense elements attributes. The first dense elements attribute is a 2-D
attribute with shape [N, ndims], which contains the indices of the elements
with nonzero values in the constant vector/tensor. The second elements
attribute is a 1-D attribute list with shape [N], which supplies the values for
each element in the first elements attribute. ndims is the rank of the
vector/tensor and N is the total nonzero elements.
The syntax is:
`sparse<` (tensor-type | vector-type)`, ` indices-attribute-list, values-attribute-list `>`
Example: a sparse tensor
sparse<vector<3x4xi32>, [[0, 0], [1, 2]], [1, 2]> represents the dense tensor
[[1, 0, 0, 0]
[0, 0, 2, 0]
[0, 0, 0, 0]]
PiperOrigin-RevId: 217764319
2018-10-19 05:02:20 +08:00
|
|
|
s << "])";
|
|
|
|
return (emitError(s.str()), nullptr);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (parseToken(Token::greater, "expected '>'"))
|
|
|
|
return nullptr;
|
|
|
|
|
|
|
|
// Build the sparse elements attribute by the indices and values.
|
|
|
|
return builder.getSparseElementsAttr(
|
2018-10-26 06:46:10 +08:00
|
|
|
type, indices.cast<DenseIntElementsAttr>(), values);
|
Add support to constant sparse tensor / vector attribute
The SparseElementsAttr uses (COO) Coordinate List encoding to represents a
sparse tensor / vector. Specifically, the coordinates and values are stored as
two dense elements attributes. The first dense elements attribute is a 2-D
attribute with shape [N, ndims], which contains the indices of the elements
with nonzero values in the constant vector/tensor. The second elements
attribute is a 1-D attribute list with shape [N], which supplies the values for
each element in the first elements attribute. ndims is the rank of the
vector/tensor and N is the total nonzero elements.
The syntax is:
`sparse<` (tensor-type | vector-type)`, ` indices-attribute-list, values-attribute-list `>`
Example: a sparse tensor
sparse<vector<3x4xi32>, [[0, 0], [1, 2]], [1, 2]> represents the dense tensor
[[1, 0, 0, 0]
[0, 0, 2, 0]
[0, 0, 0, 0]]
PiperOrigin-RevId: 217764319
2018-10-19 05:02:20 +08:00
|
|
|
}
|
|
|
|
default:
|
|
|
|
return (emitError("expected '[' to start sparse tensor literal"),
|
|
|
|
nullptr);
|
|
|
|
}
|
|
|
|
return (emitError("expected elements literal has a tensor or vector type"),
|
|
|
|
nullptr);
|
|
|
|
}
|
2018-08-03 16:54:46 +08:00
|
|
|
default: {
|
2018-10-31 05:59:22 +08:00
|
|
|
if (Type type = parseType())
|
2018-08-03 16:54:46 +08:00
|
|
|
return builder.getTypeAttr(type);
|
|
|
|
return nullptr;
|
|
|
|
}
|
|
|
|
}
|
2018-07-05 11:45:39 +08:00
|
|
|
}
|
|
|
|
|
Add support to constant sparse tensor / vector attribute
The SparseElementsAttr uses (COO) Coordinate List encoding to represents a
sparse tensor / vector. Specifically, the coordinates and values are stored as
two dense elements attributes. The first dense elements attribute is a 2-D
attribute with shape [N, ndims], which contains the indices of the elements
with nonzero values in the constant vector/tensor. The second elements
attribute is a 1-D attribute list with shape [N], which supplies the values for
each element in the first elements attribute. ndims is the rank of the
vector/tensor and N is the total nonzero elements.
The syntax is:
`sparse<` (tensor-type | vector-type)`, ` indices-attribute-list, values-attribute-list `>`
Example: a sparse tensor
sparse<vector<3x4xi32>, [[0, 0], [1, 2]], [1, 2]> represents the dense tensor
[[1, 0, 0, 0]
[0, 0, 2, 0]
[0, 0, 0, 0]]
PiperOrigin-RevId: 217764319
2018-10-19 05:02:20 +08:00
|
|
|
/// Dense elements attribute.
|
|
|
|
///
|
|
|
|
/// dense-attr-list ::= `[` attribute-value `]`
|
|
|
|
/// attribute-value ::= integer-literal
|
|
|
|
/// | float-literal
|
|
|
|
/// | `[` (attribute-value (`,` attribute-value)*)? `]`
|
|
|
|
///
|
2019-02-08 02:13:50 +08:00
|
|
|
/// This method returns a constructed dense elements attribute of tensor type
|
|
|
|
/// with the shape from the parsing result.
|
|
|
|
DenseElementsAttr Parser::parseDenseElementsAttrAsTensor(Type eltType) {
|
Add support to constant sparse tensor / vector attribute
The SparseElementsAttr uses (COO) Coordinate List encoding to represents a
sparse tensor / vector. Specifically, the coordinates and values are stored as
two dense elements attributes. The first dense elements attribute is a 2-D
attribute with shape [N, ndims], which contains the indices of the elements
with nonzero values in the constant vector/tensor. The second elements
attribute is a 1-D attribute list with shape [N], which supplies the values for
each element in the first elements attribute. ndims is the rank of the
vector/tensor and N is the total nonzero elements.
The syntax is:
`sparse<` (tensor-type | vector-type)`, ` indices-attribute-list, values-attribute-list `>`
Example: a sparse tensor
sparse<vector<3x4xi32>, [[0, 0], [1, 2]], [1, 2]> represents the dense tensor
[[1, 0, 0, 0]
[0, 0, 2, 0]
[0, 0, 0, 0]]
PiperOrigin-RevId: 217764319
2018-10-19 05:02:20 +08:00
|
|
|
TensorLiteralParser literalParser(*this, eltType);
|
|
|
|
if (literalParser.parse())
|
|
|
|
return nullptr;
|
|
|
|
|
2019-02-08 02:13:50 +08:00
|
|
|
auto type = builder.getTensorType(literalParser.getShape(), eltType);
|
2018-10-26 06:46:10 +08:00
|
|
|
return builder.getDenseElementsAttr(type, literalParser.getValues())
|
|
|
|
.cast<DenseElementsAttr>();
|
Add support to constant sparse tensor / vector attribute
The SparseElementsAttr uses (COO) Coordinate List encoding to represents a
sparse tensor / vector. Specifically, the coordinates and values are stored as
two dense elements attributes. The first dense elements attribute is a 2-D
attribute with shape [N, ndims], which contains the indices of the elements
with nonzero values in the constant vector/tensor. The second elements
attribute is a 1-D attribute list with shape [N], which supplies the values for
each element in the first elements attribute. ndims is the rank of the
vector/tensor and N is the total nonzero elements.
The syntax is:
`sparse<` (tensor-type | vector-type)`, ` indices-attribute-list, values-attribute-list `>`
Example: a sparse tensor
sparse<vector<3x4xi32>, [[0, 0], [1, 2]], [1, 2]> represents the dense tensor
[[1, 0, 0, 0]
[0, 0, 2, 0]
[0, 0, 0, 0]]
PiperOrigin-RevId: 217764319
2018-10-19 05:02:20 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
/// Dense elements attribute.
|
|
|
|
///
|
|
|
|
/// dense-attr-list ::= `[` attribute-value `]`
|
|
|
|
/// attribute-value ::= integer-literal
|
|
|
|
/// | float-literal
|
|
|
|
/// | `[` (attribute-value (`,` attribute-value)*)? `]`
|
|
|
|
///
|
|
|
|
/// This method compares the shapes from the parsing result and that from the
|
|
|
|
/// input argument. It returns a constructed dense elements attribute if both
|
|
|
|
/// match.
|
2018-10-31 05:59:22 +08:00
|
|
|
DenseElementsAttr Parser::parseDenseElementsAttr(VectorOrTensorType type) {
|
|
|
|
auto eltTy = type.getElementType();
|
2018-10-19 04:54:44 +08:00
|
|
|
TensorLiteralParser literalParser(*this, eltTy);
|
|
|
|
if (literalParser.parse())
|
|
|
|
return nullptr;
|
2018-10-31 05:59:22 +08:00
|
|
|
if (literalParser.getShape() != type.getShape()) {
|
2018-10-19 04:54:44 +08:00
|
|
|
std::string str;
|
|
|
|
llvm::raw_string_ostream s(str);
|
|
|
|
s << "inferred shape of elements literal ([";
|
|
|
|
interleaveComma(literalParser.getShape(), s);
|
|
|
|
s << "]) does not match type ([";
|
2018-10-31 05:59:22 +08:00
|
|
|
interleaveComma(type.getShape(), s);
|
2018-10-19 04:54:44 +08:00
|
|
|
s << "])";
|
|
|
|
return (emitError(s.str()), nullptr);
|
|
|
|
}
|
2018-10-26 06:46:10 +08:00
|
|
|
return builder.getDenseElementsAttr(type, literalParser.getValues())
|
|
|
|
.cast<DenseElementsAttr>();
|
2018-10-19 04:54:44 +08:00
|
|
|
}
|
|
|
|
|
Add support to constant sparse tensor / vector attribute
The SparseElementsAttr uses (COO) Coordinate List encoding to represents a
sparse tensor / vector. Specifically, the coordinates and values are stored as
two dense elements attributes. The first dense elements attribute is a 2-D
attribute with shape [N, ndims], which contains the indices of the elements
with nonzero values in the constant vector/tensor. The second elements
attribute is a 1-D attribute list with shape [N], which supplies the values for
each element in the first elements attribute. ndims is the rank of the
vector/tensor and N is the total nonzero elements.
The syntax is:
`sparse<` (tensor-type | vector-type)`, ` indices-attribute-list, values-attribute-list `>`
Example: a sparse tensor
sparse<vector<3x4xi32>, [[0, 0], [1, 2]], [1, 2]> represents the dense tensor
[[1, 0, 0, 0]
[0, 0, 2, 0]
[0, 0, 0, 0]]
PiperOrigin-RevId: 217764319
2018-10-19 05:02:20 +08:00
|
|
|
/// Vector or tensor type for elements attribute.
|
|
|
|
///
|
|
|
|
/// vector-or-tensor-type ::= vector-type | tensor-type
|
|
|
|
///
|
|
|
|
/// This method also checks the type has static shape and ranked.
|
2018-10-31 05:59:22 +08:00
|
|
|
VectorOrTensorType Parser::parseVectorOrTensorType() {
|
2018-11-10 13:24:37 +08:00
|
|
|
auto elementType = parseType();
|
|
|
|
if (!elementType)
|
|
|
|
return nullptr;
|
|
|
|
|
|
|
|
auto type = elementType.dyn_cast<VectorOrTensorType>();
|
2018-10-19 04:54:44 +08:00
|
|
|
if (!type) {
|
|
|
|
return (emitError("expected elements literal has a tensor or vector type"),
|
|
|
|
nullptr);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (parseToken(Token::comma, "expected ','"))
|
|
|
|
return nullptr;
|
|
|
|
|
2018-10-31 05:59:22 +08:00
|
|
|
if (!type.hasStaticShape() || type.getRank() == -1) {
|
2018-10-19 04:54:44 +08:00
|
|
|
return (emitError("tensor literals must be ranked and have static shape"),
|
|
|
|
nullptr);
|
|
|
|
}
|
|
|
|
return type;
|
|
|
|
}
|
|
|
|
|
2019-01-24 05:11:23 +08:00
|
|
|
/// Debug Location.
|
|
|
|
///
|
|
|
|
/// location ::= `loc` inline-location
|
|
|
|
/// inline-location ::= '(' location-inst ')'
|
|
|
|
///
|
|
|
|
ParseResult Parser::parseLocation(llvm::Optional<Location> *loc) {
|
|
|
|
assert(loc && "loc is expected to be non-null");
|
|
|
|
|
|
|
|
// Check for 'loc' identifier.
|
|
|
|
if (getToken().isNot(Token::kw_loc))
|
|
|
|
return emitError("expected location keyword");
|
|
|
|
consumeToken(Token::kw_loc);
|
|
|
|
|
|
|
|
// Parse the inline-location.
|
|
|
|
if (parseToken(Token::l_paren, "expected '(' in inline location") ||
|
|
|
|
parseLocationInstance(loc) ||
|
|
|
|
parseToken(Token::r_paren, "expected ')' in inline location"))
|
|
|
|
return ParseFailure;
|
|
|
|
return ParseSuccess;
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Specific location instances.
|
|
|
|
///
|
|
|
|
/// location-inst ::= filelinecol-location |
|
|
|
|
/// name-location |
|
|
|
|
/// callsite-location |
|
|
|
|
/// fused-location |
|
|
|
|
/// unknown-location
|
|
|
|
/// filelinecol-location ::= string-literal ':' integer-literal
|
|
|
|
/// ':' integer-literal
|
|
|
|
/// name-location ::= string-literal
|
|
|
|
/// callsite-location ::= 'callsite' '(' location-inst 'at' location-inst ')'
|
|
|
|
/// fused-location ::= fused ('<' attribute-value '>')?
|
|
|
|
/// '[' location-inst (location-inst ',')* ']'
|
|
|
|
/// unknown-location ::= 'unknown'
|
|
|
|
///
|
|
|
|
ParseResult Parser::parseLocationInstance(llvm::Optional<Location> *loc) {
|
|
|
|
auto *ctx = getContext();
|
|
|
|
|
|
|
|
// Handle either name or filelinecol locations.
|
|
|
|
if (getToken().is(Token::string)) {
|
|
|
|
auto str = getToken().getStringValue();
|
|
|
|
consumeToken(Token::string);
|
|
|
|
|
|
|
|
// If the next token is ':' this is a filelinecol location.
|
|
|
|
if (consumeIf(Token::colon)) {
|
|
|
|
// Parse the line number.
|
|
|
|
if (getToken().isNot(Token::integer))
|
|
|
|
return emitError("expected integer line number in FileLineColLoc");
|
|
|
|
auto line = getToken().getUnsignedIntegerValue();
|
|
|
|
if (!line.hasValue())
|
|
|
|
return emitError("expected integer line number in FileLineColLoc");
|
|
|
|
consumeToken(Token::integer);
|
|
|
|
|
|
|
|
// Parse the ':'.
|
|
|
|
if (parseToken(Token::colon, "expected ':' in FileLineColLoc"))
|
|
|
|
return ParseFailure;
|
|
|
|
|
|
|
|
// Parse the column number.
|
|
|
|
if (getToken().isNot(Token::integer))
|
|
|
|
return emitError("expected integer column number in FileLineColLoc");
|
|
|
|
auto column = getToken().getUnsignedIntegerValue();
|
|
|
|
if (!column.hasValue())
|
|
|
|
return emitError("expected integer column number in FileLineColLoc");
|
|
|
|
consumeToken(Token::integer);
|
|
|
|
|
|
|
|
auto file = UniquedFilename::get(str, ctx);
|
|
|
|
*loc = FileLineColLoc::get(file, line.getValue(), column.getValue(), ctx);
|
|
|
|
return ParseSuccess;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Otherwise, this is a NameLoc.
|
|
|
|
*loc = NameLoc::get(Identifier::get(str, ctx), ctx);
|
|
|
|
return ParseSuccess;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Check for a 'unknown' for an unknown location.
|
|
|
|
if (getToken().is(Token::bare_identifier) &&
|
|
|
|
getToken().getSpelling() == "unknown") {
|
|
|
|
consumeToken(Token::bare_identifier);
|
|
|
|
*loc = UnknownLoc::get(ctx);
|
|
|
|
return ParseSuccess;
|
|
|
|
}
|
|
|
|
|
|
|
|
// If the token is 'fused', then this is a fused location.
|
|
|
|
if (getToken().is(Token::bare_identifier) &&
|
|
|
|
getToken().getSpelling() == "fused") {
|
|
|
|
consumeToken(Token::bare_identifier);
|
|
|
|
|
|
|
|
// Try to parse the optional metadata.
|
|
|
|
Attribute metadata;
|
|
|
|
if (consumeIf(Token::less)) {
|
|
|
|
metadata = parseAttribute();
|
|
|
|
if (!metadata)
|
|
|
|
return emitError("expected valid attribute metadata");
|
|
|
|
// Parse the '>' token.
|
|
|
|
if (parseToken(Token::greater,
|
|
|
|
"expected '>' after fused location metadata"))
|
|
|
|
return ParseFailure;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Parse the '['.
|
|
|
|
if (parseToken(Token::l_square, "expected '[' in fused location"))
|
|
|
|
return ParseFailure;
|
|
|
|
|
|
|
|
// Parse the internal locations.
|
|
|
|
llvm::SmallVector<Location, 4> locations;
|
|
|
|
do {
|
|
|
|
llvm::Optional<Location> newLoc;
|
|
|
|
if (parseLocationInstance(&newLoc))
|
|
|
|
return ParseFailure;
|
|
|
|
locations.push_back(*newLoc);
|
|
|
|
|
|
|
|
// Parse the ','.
|
|
|
|
} while (consumeIf(Token::comma));
|
|
|
|
|
|
|
|
// Parse the ']'.
|
|
|
|
if (parseToken(Token::r_square, "expected ']' in fused location"))
|
|
|
|
return ParseFailure;
|
|
|
|
|
|
|
|
// Return the fused location.
|
|
|
|
if (metadata)
|
|
|
|
*loc = FusedLoc::get(locations, metadata, getContext());
|
|
|
|
else
|
|
|
|
*loc = FusedLoc::get(locations, ctx);
|
|
|
|
return ParseSuccess;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Check for the 'callsite' signifying a callsite location.
|
|
|
|
if (getToken().is(Token::bare_identifier) &&
|
|
|
|
getToken().getSpelling() == "callsite") {
|
|
|
|
consumeToken(Token::bare_identifier);
|
|
|
|
|
|
|
|
// Parse the '('.
|
|
|
|
if (parseToken(Token::l_paren, "expected '(' in callsite location"))
|
|
|
|
return ParseFailure;
|
|
|
|
|
|
|
|
// Parse the callee location.
|
|
|
|
llvm::Optional<Location> calleeLoc;
|
|
|
|
if (parseLocationInstance(&calleeLoc))
|
|
|
|
return ParseFailure;
|
|
|
|
|
|
|
|
// Parse the 'at'.
|
|
|
|
if (getToken().isNot(Token::bare_identifier) ||
|
|
|
|
getToken().getSpelling() != "at")
|
|
|
|
return emitError("expected 'at' in callsite location");
|
|
|
|
consumeToken(Token::bare_identifier);
|
|
|
|
|
|
|
|
// Parse the caller location.
|
|
|
|
llvm::Optional<Location> callerLoc;
|
|
|
|
if (parseLocationInstance(&callerLoc))
|
|
|
|
return ParseFailure;
|
|
|
|
|
|
|
|
// Parse the ')'.
|
|
|
|
if (parseToken(Token::r_paren, "expected ')' in callsite location"))
|
|
|
|
return ParseFailure;
|
|
|
|
|
|
|
|
// Return the callsite location.
|
|
|
|
*loc = CallSiteLoc::get(*calleeLoc, *callerLoc, ctx);
|
|
|
|
return ParseSuccess;
|
|
|
|
}
|
|
|
|
|
|
|
|
return emitError("expected location instance");
|
|
|
|
}
|
|
|
|
|
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-10-14 22:55:29 +08:00
|
|
|
if (!consumeIf(Token::l_brace))
|
|
|
|
return ParseFailure;
|
2018-07-05 11:45:39 +08:00
|
|
|
|
|
|
|
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-10-19 04:54:44 +08:00
|
|
|
/// Higher precedence ops - all at the same precedence level. HNoOp is false
|
|
|
|
/// in the boolean sense.
|
2018-07-11 01:08:27 +08:00
|
|
|
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
|
|
|
|
2019-01-27 02:41:17 +08:00
|
|
|
AffineMap parseAffineMapInline();
|
2018-10-26 09:33:42 +08:00
|
|
|
AffineMap parseAffineMapRange(unsigned numDims, unsigned numSymbols);
|
2019-01-27 02:41:17 +08:00
|
|
|
IntegerSet parseIntegerSetInline();
|
|
|
|
ParseResult parseAffineMapOrIntegerSetInline(AffineMap &map, IntegerSet &set);
|
2018-10-26 09:33:42 +08:00
|
|
|
IntegerSet parseIntegerSetConstraints(unsigned numDims, unsigned numSymbols);
|
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);
|
2019-01-27 02:41:17 +08:00
|
|
|
ParseResult parseDimAndOptionalSymbolIdList(unsigned &numDims,
|
|
|
|
unsigned &numSymbols);
|
2018-10-09 04:47:18 +08:00
|
|
|
ParseResult parseIdentifierDefinition(AffineExpr idExpr);
|
|
|
|
|
|
|
|
AffineExpr parseAffineExpr();
|
|
|
|
AffineExpr parseParentheticalExpr();
|
|
|
|
AffineExpr parseNegateExpression(AffineExpr lhs);
|
|
|
|
AffineExpr parseIntegerExpr();
|
|
|
|
AffineExpr parseBareIdExpr();
|
|
|
|
|
2019-01-07 23:51:23 +08:00
|
|
|
AffineExpr getAffineBinaryOpExpr(AffineHighPrecOp op, AffineExpr lhs,
|
2018-10-09 04:47:18 +08:00
|
|
|
AffineExpr rhs, SMLoc opLoc);
|
2019-01-07 23:51:23 +08:00
|
|
|
AffineExpr getAffineBinaryOpExpr(AffineLowPrecOp op, AffineExpr lhs,
|
2018-10-09 04:47:18 +08:00
|
|
|
AffineExpr rhs);
|
|
|
|
AffineExpr parseAffineOperandExpr(AffineExpr lhs);
|
|
|
|
AffineExpr parseAffineLowPrecOpExpr(AffineExpr llhs, AffineLowPrecOp llhsOp);
|
|
|
|
AffineExpr parseAffineHighPrecOpExpr(AffineExpr llhs, AffineHighPrecOp llhsOp,
|
|
|
|
SMLoc llhsOpLoc);
|
|
|
|
AffineExpr parseAffineConstraint(bool *isEq);
|
2018-07-11 01:08:27 +08:00
|
|
|
|
|
|
|
private:
|
2018-10-09 04:47:18 +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.
|
2019-01-07 23:51:23 +08:00
|
|
|
AffineExpr AffineParser::getAffineBinaryOpExpr(AffineHighPrecOp op,
|
2018-10-09 04:47:18 +08:00
|
|
|
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-10-10 01:59:27 +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-10-10 01:59:27 +08:00
|
|
|
return lhs * rhs;
|
2018-07-04 11:16:08 +08:00
|
|
|
case FloorDiv:
|
2018-10-10 01:59:27 +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-10-10 01:59:27 +08:00
|
|
|
return lhs.floorDiv(rhs);
|
2018-07-04 11:16:08 +08:00
|
|
|
case CeilDiv:
|
2018-10-10 01:59:27 +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-10-10 01:59:27 +08:00
|
|
|
return lhs.ceilDiv(rhs);
|
2018-07-04 11:16:08 +08:00
|
|
|
case Mod:
|
2018-10-10 01:59:27 +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-10-10 01:59:27 +08:00
|
|
|
return 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).
|
2019-01-07 23:51:23 +08:00
|
|
|
AffineExpr AffineParser::getAffineBinaryOpExpr(AffineLowPrecOp op,
|
2018-10-09 04:47:18 +08:00
|
|
|
AffineExpr lhs, AffineExpr rhs) {
|
2018-07-04 11:16:08 +08:00
|
|
|
switch (op) {
|
|
|
|
case AffineLowPrecOp::Add:
|
2018-10-10 01:59:27 +08:00
|
|
|
return lhs + rhs;
|
2018-07-04 11:16:08 +08:00
|
|
|
case AffineLowPrecOp::Sub:
|
2018-10-10 01:59:27 +08:00
|
|
|
return lhs - rhs;
|
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;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-10-19 04:54:44 +08:00
|
|
|
/// Consume this token if it is a lower precedence affine op (there are only
|
|
|
|
/// two 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-10-09 04:47:18 +08:00
|
|
|
AffineExpr AffineParser::parseAffineHighPrecOpExpr(AffineExpr llhs,
|
|
|
|
AffineHighPrecOp llhsOp,
|
|
|
|
SMLoc llhsOpLoc) {
|
|
|
|
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) {
|
2019-01-07 23:51:23 +08:00
|
|
|
AffineExpr expr = getAffineBinaryOpExpr(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)
|
2019-01-07 23:51:23 +08:00
|
|
|
return getAffineBinaryOpExpr(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-10-09 04:47:18 +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-10-08 23:09:50 +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-10-09 04:47:18 +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-10-09 04:47:18 +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-10-10 01:59:27 +08:00
|
|
|
return (-1) * 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-10-09 04:47:18 +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-10-09 04:47:18 +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-10-07 08:21:53 +08:00
|
|
|
return (emitError("constant too large for index"), nullptr);
|
2018-08-21 23:42:19 +08:00
|
|
|
|
2018-07-10 00:00:25 +08:00
|
|
|
consumeToken(Token::integer);
|
2018-10-09 01:20:25 +08:00
|
|
|
return builder.getAffineConstantExpr((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
|
2018-10-19 04:54:44 +08:00
|
|
|
// parseAffineHighPrecOpExpression(). However, for i + (j*k) + -l, (j*k) and
|
|
|
|
// -l are valid operands that will be parsed by this function.
|
2018-10-09 04:47:18 +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,
|
2018-10-19 04:54:44 +08:00
|
|
|
/// 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-10-19 04:54:44 +08:00
|
|
|
/// will return the affine expr equivalent of (e1 + (e2*e3)) + e4, where
|
|
|
|
/// (e2*e3) will be parsed using parseAffineHighPrecOpExpr().
|
2018-10-09 04:47:18 +08:00
|
|
|
AffineExpr AffineParser::parseAffineLowPrecOpExpr(AffineExpr llhs,
|
|
|
|
AffineLowPrecOp llhsOp) {
|
|
|
|
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) {
|
2019-01-07 23:51:23 +08:00
|
|
|
AffineExpr sum = getAffineBinaryOpExpr(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-10-09 04:47:18 +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-10-09 04:47:18 +08:00
|
|
|
AffineExpr expr =
|
2019-01-07 23:51:23 +08:00
|
|
|
llhs ? getAffineBinaryOpExpr(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)
|
2019-01-07 23:51:23 +08:00
|
|
|
return getAffineBinaryOpExpr(llhsOp, llhs, lhs);
|
2018-07-10 00:00:25 +08:00
|
|
|
// 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
|
|
|
|
///
|
2018-10-19 04:54:44 +08:00
|
|
|
/// Additional conditions are checked depending on the production. For eg.,
|
|
|
|
/// one of the operands for `*` has to be either constant/symbolic; the second
|
2018-07-10 00:00:25 +08:00
|
|
|
/// operand for floordiv, ceildiv, and mod has to be a positive integer.
|
2018-10-09 04:47:18 +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-10-19 04:54:44 +08:00
|
|
|
/// Parse a dim or symbol from the lists appearing before the actual
|
|
|
|
/// expressions of the affine map. Update our state to store the
|
|
|
|
/// dimensional/symbolic identifier.
|
2018-10-09 04:47:18 +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 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,
|
2019-01-27 02:41:17 +08:00
|
|
|
"expected '(' at start of dimensional identifiers list")) {
|
2018-07-24 08:30:01 +08:00
|
|
|
return ParseFailure;
|
2019-01-27 02:41:17 +08:00
|
|
|
}
|
2018-06-30 09:09:29 +08:00
|
|
|
|
2018-07-26 03:55:50 +08:00
|
|
|
auto parseElt = [&]() -> ParseResult {
|
2018-10-09 01:20:25 +08:00
|
|
|
auto dimension = getAffineDimExpr(numDims++, getContext());
|
2018-07-26 03:55:50 +08:00
|
|
|
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
|
|
|
}
|
|
|
|
|
2019-01-27 02:41:17 +08:00
|
|
|
/// Parse the list of symbolic identifiers to an affine map.
|
|
|
|
ParseResult AffineParser::parseSymbolIdList(unsigned &numSymbols) {
|
|
|
|
consumeToken(Token::l_square);
|
|
|
|
auto parseElt = [&]() -> ParseResult {
|
|
|
|
auto symbol = getAffineSymbolExpr(numSymbols++, getContext());
|
|
|
|
return parseIdentifierDefinition(symbol);
|
|
|
|
};
|
|
|
|
return parseCommaSeparatedListUntil(Token::r_square, parseElt);
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Parse the list of symbolic identifiers to an affine map.
|
|
|
|
ParseResult
|
|
|
|
AffineParser::parseDimAndOptionalSymbolIdList(unsigned &numDims,
|
|
|
|
unsigned &numSymbols) {
|
|
|
|
if (parseDimIdList(numDims)) {
|
|
|
|
return ParseResult::ParseFailure;
|
|
|
|
}
|
|
|
|
if (!getToken().is(Token::l_square)) {
|
|
|
|
numSymbols = 0;
|
|
|
|
return ParseResult::ParseSuccess;
|
|
|
|
}
|
|
|
|
return parseSymbolIdList(numSymbols);
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Parses an affine map definition inline.
|
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-10-26 09:33:42 +08:00
|
|
|
///
|
2019-01-27 02:41:17 +08:00
|
|
|
AffineMap AffineParser::parseAffineMapInline() {
|
|
|
|
unsigned numDims = 0, numSymbols = 0;
|
|
|
|
|
|
|
|
// List of dimensional and optional symbol identifiers.
|
|
|
|
if (parseDimAndOptionalSymbolIdList(numDims, numSymbols)) {
|
|
|
|
return AffineMap();
|
|
|
|
}
|
|
|
|
|
|
|
|
if (parseToken(Token::arrow, "expected '->' or '['")) {
|
|
|
|
return AffineMap();
|
|
|
|
}
|
|
|
|
|
|
|
|
// Parse the affine map.
|
|
|
|
return parseAffineMapRange(numDims, numSymbols);
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Parses an integer set definition inline.
|
2018-10-26 09:33:42 +08:00
|
|
|
///
|
|
|
|
/// integer-set-inline
|
|
|
|
/// ::= dim-and-symbol-id-lists `:`
|
|
|
|
/// affine-constraint-conjunction
|
|
|
|
/// affine-constraint-conjunction ::= /*empty*/
|
|
|
|
/// | affine-constraint (`,`
|
|
|
|
/// affine-constraint)*
|
|
|
|
///
|
2019-01-27 02:41:17 +08:00
|
|
|
IntegerSet AffineParser::parseIntegerSetInline() {
|
2018-07-26 03:55:50 +08:00
|
|
|
unsigned numDims = 0, numSymbols = 0;
|
|
|
|
|
2019-01-27 02:41:17 +08:00
|
|
|
// List of dimensional and optional symbol identifiers.
|
|
|
|
if (parseDimAndOptionalSymbolIdList(numDims, numSymbols)) {
|
|
|
|
return IntegerSet();
|
2018-10-26 09:33:42 +08:00
|
|
|
}
|
2018-06-30 09:09:29 +08:00
|
|
|
|
2019-01-27 02:41:17 +08:00
|
|
|
if (parseToken(Token::colon, "expected ':' or '['")) {
|
|
|
|
return IntegerSet();
|
|
|
|
}
|
|
|
|
|
|
|
|
return parseIntegerSetConstraints(numDims, numSymbols);
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Parses an ambiguous affine map or integer set definition inline.
|
|
|
|
ParseResult AffineParser::parseAffineMapOrIntegerSetInline(AffineMap &map,
|
|
|
|
IntegerSet &set) {
|
|
|
|
unsigned numDims = 0, numSymbols = 0;
|
|
|
|
|
|
|
|
// List of dimensional and optional symbol identifiers.
|
|
|
|
if (parseDimAndOptionalSymbolIdList(numDims, numSymbols)) {
|
|
|
|
return ParseResult::ParseFailure;
|
2018-06-30 09:09:29 +08:00
|
|
|
}
|
2018-07-24 08:30:01 +08:00
|
|
|
|
2018-10-26 09:33:42 +08:00
|
|
|
// This is needed for parsing attributes as we wouldn't know whether we would
|
|
|
|
// be parsing an integer set attribute or an affine map attribute.
|
2019-01-27 02:41:17 +08:00
|
|
|
bool isArrow = getToken().is(Token::arrow);
|
|
|
|
bool isColon = getToken().is(Token::colon);
|
|
|
|
if (!isArrow && !isColon) {
|
2019-02-02 16:07:30 +08:00
|
|
|
return emitError("expected '->' or ':'");
|
2019-01-27 02:41:17 +08:00
|
|
|
} else if (isArrow) {
|
|
|
|
parseToken(Token::arrow, "expected '->' or '['");
|
|
|
|
map = parseAffineMapRange(numDims, numSymbols);
|
|
|
|
return map ? ParseSuccess : ParseFailure;
|
|
|
|
} else if (parseToken(Token::colon, "expected ':' or '['")) {
|
|
|
|
return ParseFailure;
|
2018-10-26 09:33:42 +08:00
|
|
|
}
|
2019-01-27 02:41:17 +08:00
|
|
|
|
|
|
|
if ((set = parseIntegerSetConstraints(numDims, numSymbols)))
|
|
|
|
return ParseSuccess;
|
|
|
|
|
|
|
|
return ParseFailure;
|
2018-10-26 09:33:42 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
/// Parse the range and sizes affine map definition inline.
|
|
|
|
///
|
|
|
|
/// 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)+ `)`
|
|
|
|
///
|
|
|
|
/// multi-dim-affine-expr ::= `(` affine-expr (`,` affine-expr)* `)
|
|
|
|
AffineMap AffineParser::parseAffineMapRange(unsigned numDims,
|
|
|
|
unsigned numSymbols) {
|
|
|
|
parseToken(Token::l_paren, "expected '(' at start of affine map range");
|
2018-06-30 09:09:29 +08:00
|
|
|
|
2018-10-09 04:47:18 +08:00
|
|
|
SmallVector<AffineExpr, 4> exprs;
|
2018-06-30 09:09:29 +08:00
|
|
|
auto parseElt = [&]() -> ParseResult {
|
2018-10-08 23:09:50 +08:00
|
|
|
auto elt = parseAffineExpr();
|
2018-06-30 09:09:29 +08:00
|
|
|
ParseResult res = elt ? ParseSuccess : ParseFailure;
|
|
|
|
exprs.push_back(elt);
|
|
|
|
return res;
|
|
|
|
};
|
|
|
|
|
2018-10-19 04:54:44 +08:00
|
|
|
// Parse a multi-dimensional affine expression (a comma-separated list of
|
|
|
|
// 1-d 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))
|
2019-01-27 02:41:17 +08:00
|
|
|
return AffineMap();
|
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.
|
2018-10-09 04:47:18 +08:00
|
|
|
SmallVector<AffineExpr, 4> rangeSizes;
|
2018-07-12 12:31:07 +08:00
|
|
|
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"))
|
2019-01-27 02:41:17 +08:00
|
|
|
return AffineMap();
|
2018-07-12 12:31:07 +08:00
|
|
|
|
|
|
|
auto parseRangeSize = [&]() -> ParseResult {
|
2018-07-26 03:55:50 +08:00
|
|
|
auto loc = getToken().getLoc();
|
2018-10-08 23:09:50 +08:00
|
|
|
auto elt = parseAffineExpr();
|
2018-07-26 03:55:50 +08:00
|
|
|
if (!elt)
|
|
|
|
return ParseFailure;
|
|
|
|
|
2018-10-10 01:59:27 +08:00
|
|
|
if (!elt.isSymbolicOrConstant())
|
2018-07-26 03:55:50 +08:00
|
|
|
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))
|
2019-01-27 02:41:17 +08:00
|
|
|
return AffineMap();
|
2018-07-12 12:31:07 +08:00
|
|
|
if (exprs.size() > rangeSizes.size())
|
|
|
|
return (emitError(loc, "fewer range sizes than range expressions"),
|
2019-01-27 02:41:17 +08:00
|
|
|
AffineMap());
|
2018-07-12 12:31:07 +08:00
|
|
|
if (exprs.size() < rangeSizes.size())
|
|
|
|
return (emitError(loc, "more range sizes than range expressions"),
|
2019-01-27 02:41:17 +08:00
|
|
|
AffineMap());
|
2018-07-12 12:31:07 +08:00
|
|
|
}
|
|
|
|
|
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
|
|
|
}
|
|
|
|
|
2019-01-27 02:41:17 +08:00
|
|
|
/// 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().isNot(Token::hash_identifier)) {
|
|
|
|
// Try to parse inline integer set.
|
|
|
|
return AffineParser(state).parseIntegerSetInline();
|
|
|
|
}
|
2018-10-26 09:33:42 +08:00
|
|
|
|
2019-01-27 02:41:17 +08:00
|
|
|
// Parse integer set identifier and verify that it exists.
|
|
|
|
StringRef id = getTokenSpelling().drop_front();
|
|
|
|
if (getState().integerSetDefinitions.count(id) > 0) {
|
|
|
|
consumeToken(Token::hash_identifier);
|
|
|
|
return getState().integerSetDefinitions[id];
|
|
|
|
}
|
|
|
|
|
|
|
|
// The id isn't among any of the recorded definitions.
|
|
|
|
emitError("undefined integer set id '" + id + "'");
|
|
|
|
return IntegerSet();
|
2018-07-11 01:08:27 +08:00
|
|
|
}
|
|
|
|
|
2019-01-27 02:41:17 +08:00
|
|
|
/// Parse a reference to an affine map.
|
|
|
|
/// affine-map ::= affine-map-id | affine-map-inline
|
|
|
|
/// affine-map-id ::= `#` suffix-id
|
2018-10-26 13:13:03 +08:00
|
|
|
///
|
2019-01-27 02:41:17 +08:00
|
|
|
AffineMap Parser::parseAffineMapReference() {
|
|
|
|
if (getToken().isNot(Token::hash_identifier)) {
|
|
|
|
// Try to parse inline affine map.
|
|
|
|
return AffineParser(state).parseAffineMapInline();
|
|
|
|
}
|
|
|
|
|
|
|
|
// Parse affine map identifier and verify that it exists.
|
|
|
|
StringRef id = getTokenSpelling().drop_front();
|
|
|
|
if (getState().affineMapDefinitions.count(id) > 0) {
|
|
|
|
consumeToken(Token::hash_identifier);
|
|
|
|
return getState().affineMapDefinitions[id];
|
|
|
|
}
|
|
|
|
|
|
|
|
// The id isn't among any of the recorded definitions.
|
|
|
|
emitError("undefined affine map id '" + id + "'");
|
|
|
|
return AffineMap();
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Parse an ambiguous reference to either and affine map or an integer set.
|
|
|
|
ParseResult Parser::parseAffineMapOrIntegerSetReference(AffineMap &map,
|
|
|
|
IntegerSet &set) {
|
2018-10-26 13:13:03 +08:00
|
|
|
if (getToken().isNot(Token::hash_identifier)) {
|
2019-01-27 02:41:17 +08:00
|
|
|
// Try to parse inline affine map.
|
|
|
|
return AffineParser(state).parseAffineMapOrIntegerSetInline(map, set);
|
2018-10-26 13:13:03 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
// Parse affine map / integer set identifier and verify that it exists.
|
|
|
|
// Note that an id can't be in both affineMapDefinitions and
|
|
|
|
// integerSetDefinitions since they use the same sigil '#'.
|
2019-01-27 02:41:17 +08:00
|
|
|
StringRef id = getTokenSpelling().drop_front();
|
|
|
|
if (getState().affineMapDefinitions.count(id) > 0) {
|
2018-10-26 13:13:03 +08:00
|
|
|
consumeToken(Token::hash_identifier);
|
2019-01-27 02:41:17 +08:00
|
|
|
map = getState().affineMapDefinitions[id];
|
|
|
|
return ParseSuccess;
|
2018-10-26 13:13:03 +08:00
|
|
|
}
|
2019-01-27 02:41:17 +08:00
|
|
|
if (getState().integerSetDefinitions.count(id) > 0) {
|
2018-07-17 00:45:22 +08:00
|
|
|
consumeToken(Token::hash_identifier);
|
2019-01-27 02:41:17 +08:00
|
|
|
set = getState().integerSetDefinitions[id];
|
|
|
|
return ParseSuccess;
|
2018-07-17 00:45:22 +08:00
|
|
|
}
|
2018-10-26 13:13:03 +08:00
|
|
|
|
|
|
|
// The id isn't among any of the recorded definitions.
|
2019-01-27 02:41:17 +08:00
|
|
|
emitError("undefined affine map or integer set id '" + id + "'");
|
2018-10-26 13:13:03 +08:00
|
|
|
|
2019-01-27 02:41:17 +08:00
|
|
|
return ParseFailure;
|
2018-07-17 00:45:22 +08:00
|
|
|
}
|
|
|
|
|
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 {
|
2018-10-19 04:54:44 +08:00
|
|
|
/// This class contains parser state that is common across CFG and ML
|
|
|
|
/// functions, notably for dealing with operations and SSA values.
|
2018-07-19 23:35:28 +08:00
|
|
|
class FunctionParser : public Parser {
|
|
|
|
public:
|
2018-12-29 10:41:31 +08:00
|
|
|
/// This builder intentionally shadows the builder in the base class, with a
|
|
|
|
/// more specific builder type.
|
|
|
|
#pragma clang diagnostic push
|
|
|
|
#pragma clang diagnostic ignored "-Wshadow-field"
|
|
|
|
FuncBuilder builder;
|
|
|
|
#pragma clang diagnostic pop
|
|
|
|
|
|
|
|
FunctionParser(ParserState &state, Function *function)
|
|
|
|
: Parser(state), builder(function), function(function) {}
|
|
|
|
|
|
|
|
~FunctionParser();
|
|
|
|
|
2018-12-30 01:01:59 +08:00
|
|
|
ParseResult parseFunctionBody(bool hadNamedArguments);
|
2018-12-29 10:41:31 +08:00
|
|
|
|
|
|
|
/// Parse a single operation successor and it's operand list.
|
|
|
|
bool parseSuccessorAndUseList(Block *&dest,
|
|
|
|
SmallVectorImpl<Value *> &operands);
|
|
|
|
|
2019-01-10 04:28:30 +08:00
|
|
|
/// Parse a comma-separated list of operation successors in brackets.
|
|
|
|
ParseResult
|
|
|
|
parseSuccessors(SmallVectorImpl<Block *> &destinations,
|
|
|
|
SmallVectorImpl<SmallVector<Value *, 4>> &operands);
|
|
|
|
|
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-12-29 10:41:31 +08:00
|
|
|
ParseResult finalizeFunction(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
|
2018-10-19 04:54:44 +08:00
|
|
|
/// third is the location of the reference, which is used in case this ends
|
|
|
|
/// up being a use of an undefined value.
|
2018-07-21 09:41:34 +08:00
|
|
|
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
|
|
|
|
2018-10-19 04:54:44 +08:00
|
|
|
/// Given a reference to an SSA value and its type, return a reference. This
|
2018-07-19 23:35:28 +08:00
|
|
|
/// returns null on failure.
|
2018-12-28 06:35:10 +08:00
|
|
|
Value *resolveSSAUse(SSAUseInfo useInfo, Type type);
|
2018-07-19 23:35:28 +08:00
|
|
|
|
|
|
|
/// Register a definition of a value with the symbol table.
|
2018-12-28 06:35:10 +08:00
|
|
|
ParseResult addDefinition(SSAUseInfo useInfo, Value *value);
|
2018-07-19 23:35:28 +08:00
|
|
|
|
|
|
|
// 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(
|
2018-10-31 05:59:22 +08:00
|
|
|
const std::function<ResultType(SSAUseInfo, Type)> &action);
|
2018-07-23 06:45:24 +08:00
|
|
|
|
2018-12-28 06:35:10 +08:00
|
|
|
Value *parseSSAUseAndType() {
|
|
|
|
return parseSSADefOrUseAndType<Value *>(
|
|
|
|
[&](SSAUseInfo useInfo, Type type) -> Value * {
|
2018-07-23 06:45:24 +08:00
|
|
|
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-11-13 06:58:53 +08:00
|
|
|
parseOptionalSSAUseAndTypeList(SmallVectorImpl<ValueTy *> &results);
|
2018-07-19 23:35:28 +08:00
|
|
|
|
2018-12-29 10:41:31 +08:00
|
|
|
// Block references.
|
2018-07-19 23:35:28 +08:00
|
|
|
|
2019-02-02 08:42:18 +08:00
|
|
|
ParseResult
|
|
|
|
parseOperationBlockList(SmallVectorImpl<Block *> &results,
|
|
|
|
ArrayRef<std::pair<SSAUseInfo, Type>> entryArguments);
|
|
|
|
ParseResult parseBlockListBody(SmallVectorImpl<Block *> &results);
|
|
|
|
ParseResult parseBlock(Block *&block);
|
|
|
|
ParseResult parseBlockBody(Block *block);
|
|
|
|
|
|
|
|
ParseResult
|
|
|
|
parseOptionalBlockArgList(SmallVectorImpl<BlockArgument *> &results,
|
|
|
|
Block *owner);
|
|
|
|
|
|
|
|
/// Cleans up the memory for allocated blocks when a parser error occurs.
|
|
|
|
void cleanupInvalidBlocks(ArrayRef<Block *> invalidBlocks) {
|
|
|
|
// Add the referenced blocks to the function so that they can be properly
|
|
|
|
// cleaned up when the function is destroyed.
|
|
|
|
for (auto *block : invalidBlocks)
|
|
|
|
function->push_back(block);
|
|
|
|
}
|
|
|
|
|
2018-12-29 13:24:30 +08:00
|
|
|
/// Get the block with the specified name, creating it if it doesn't
|
2018-12-29 10:41:31 +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.
|
|
|
|
Block *getBlockNamed(StringRef name, SMLoc loc);
|
2018-11-16 01:56:06 +08:00
|
|
|
|
2018-12-30 01:01:59 +08:00
|
|
|
// Define the block with the specified name. Returns the Block* or
|
2018-12-29 10:41:31 +08:00
|
|
|
// nullptr in the case of redefinition.
|
2018-12-30 05:36:59 +08:00
|
|
|
Block *defineBlockNamed(StringRef name, SMLoc loc, Block *existing);
|
2018-07-27 09:09:20 +08:00
|
|
|
|
2018-12-29 10:41:31 +08:00
|
|
|
// Operations
|
|
|
|
ParseResult parseOperation();
|
2019-02-05 02:34:20 +08:00
|
|
|
Instruction *parseGenericOperation();
|
|
|
|
Instruction *parseCustomOperation();
|
2018-12-29 10:41:31 +08:00
|
|
|
|
|
|
|
ParseResult parseInstructions(Block *block);
|
2018-10-14 22:55:29 +08:00
|
|
|
|
2018-07-19 23:35:28 +08:00
|
|
|
private:
|
2018-12-29 10:41:31 +08:00
|
|
|
Function *function;
|
|
|
|
|
|
|
|
// This keeps track of the block names as well as the location of the first
|
|
|
|
// reference, used to diagnose invalid block references and memoize them.
|
|
|
|
llvm::StringMap<std::pair<Block *, SMLoc>> blocksByName;
|
|
|
|
DenseMap<Block *, SMLoc> forwardRef;
|
|
|
|
|
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.
|
2018-12-28 06:35:10 +08:00
|
|
|
llvm::StringMap<SmallVector<std::pair<Value *, SMLoc>, 1>> values;
|
2018-07-21 09:41:34 +08:00
|
|
|
|
|
|
|
/// 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.
|
2018-12-28 06:35:10 +08:00
|
|
|
DenseMap<Value *, SMLoc> forwardReferencePlaceholders;
|
2018-07-21 09:41:34 +08:00
|
|
|
|
2018-12-28 06:35:10 +08:00
|
|
|
Value *createForwardReferencePlaceholder(SMLoc loc, Type type);
|
2018-07-21 09:41:34 +08:00
|
|
|
|
|
|
|
/// Return true if this is a forward reference.
|
2018-12-28 06:35:10 +08:00
|
|
|
bool isForwardReferencePlaceholder(Value *value) {
|
2018-07-21 09:41:34 +08:00
|
|
|
return forwardReferencePlaceholders.count(value);
|
|
|
|
}
|
2018-07-19 23:35:28 +08:00
|
|
|
};
|
|
|
|
} // end anonymous namespace
|
|
|
|
|
2018-12-30 01:01:59 +08:00
|
|
|
ParseResult FunctionParser::parseFunctionBody(bool hadNamedArguments) {
|
2018-12-29 10:41:31 +08:00
|
|
|
auto braceLoc = getToken().getLoc();
|
|
|
|
if (parseToken(Token::l_brace, "expected '{' in function"))
|
|
|
|
return ParseFailure;
|
|
|
|
|
|
|
|
// Make sure we have at least one block.
|
|
|
|
if (getToken().is(Token::r_brace))
|
|
|
|
return emitError("function must have a body");
|
|
|
|
|
2018-12-30 05:36:59 +08:00
|
|
|
// If we had named arguments, then we don't allow a block name.
|
2018-12-30 01:01:59 +08:00
|
|
|
if (hadNamedArguments) {
|
2018-12-30 05:36:59 +08:00
|
|
|
if (getToken().is(Token::caret_identifier))
|
|
|
|
return emitError("invalid block name in function with named arguments");
|
2018-12-29 10:41:31 +08:00
|
|
|
}
|
|
|
|
|
2018-12-30 05:36:59 +08:00
|
|
|
// The first block is already created and should be filled in.
|
|
|
|
auto firstBlock = &function->front();
|
|
|
|
|
2019-01-26 04:48:25 +08:00
|
|
|
// Parse the first block.
|
|
|
|
if (parseBlock(firstBlock))
|
|
|
|
return ParseFailure;
|
2018-12-30 01:01:59 +08:00
|
|
|
|
2019-01-26 04:48:25 +08:00
|
|
|
// Parse the remaining list of blocks.
|
|
|
|
SmallVector<Block *, 16> blocks;
|
|
|
|
if (parseBlockListBody(blocks))
|
|
|
|
return ParseFailure;
|
|
|
|
function->getBlocks().insert(function->end(), blocks.begin(), blocks.end());
|
2018-12-30 05:36:59 +08:00
|
|
|
|
2018-12-29 10:41:31 +08:00
|
|
|
// Verify that all referenced blocks were defined.
|
|
|
|
if (!forwardRef.empty()) {
|
|
|
|
SmallVector<std::pair<const char *, Block *>, 4> errors;
|
|
|
|
// Iteration over the map isn't deterministic, so sort by source location.
|
2019-01-26 04:48:25 +08:00
|
|
|
for (auto entry : forwardRef) {
|
2018-12-29 10:41:31 +08:00
|
|
|
errors.push_back({entry.second.getPointer(), entry.first});
|
2019-01-26 04:48:25 +08:00
|
|
|
cleanupInvalidBlocks(entry.first);
|
|
|
|
}
|
2018-12-29 10:41:31 +08:00
|
|
|
llvm::array_pod_sort(errors.begin(), errors.end());
|
|
|
|
|
|
|
|
for (auto entry : errors) {
|
|
|
|
auto loc = SMLoc::getFromPointer(entry.first);
|
2018-12-30 01:01:59 +08:00
|
|
|
emitError(loc, "reference to an undefined block");
|
2018-12-29 10:41:31 +08:00
|
|
|
}
|
|
|
|
return ParseFailure;
|
|
|
|
}
|
|
|
|
|
|
|
|
return finalizeFunction(braceLoc);
|
|
|
|
}
|
|
|
|
|
2019-01-26 04:48:25 +08:00
|
|
|
/// Block list.
|
|
|
|
///
|
|
|
|
/// block-list ::= '{' block-list-body
|
|
|
|
///
|
2019-02-02 08:42:18 +08:00
|
|
|
ParseResult FunctionParser::parseOperationBlockList(
|
|
|
|
SmallVectorImpl<Block *> &results,
|
|
|
|
ArrayRef<std::pair<FunctionParser::SSAUseInfo, Type>> entryArguments) {
|
2019-01-26 04:48:25 +08:00
|
|
|
// Parse the '{'.
|
|
|
|
if (parseToken(Token::l_brace, "expected '{' to begin block list"))
|
|
|
|
return ParseFailure;
|
2019-02-02 08:42:18 +08:00
|
|
|
|
2019-01-26 04:48:25 +08:00
|
|
|
// Check for an empty block list.
|
2019-02-02 08:42:18 +08:00
|
|
|
if (entryArguments.empty() && consumeIf(Token::r_brace))
|
2019-01-26 04:48:25 +08:00
|
|
|
return ParseSuccess;
|
|
|
|
Block *currentBlock = builder.getInsertionBlock();
|
|
|
|
|
|
|
|
// Parse the first block directly to allow for it to be unnamed.
|
|
|
|
Block *block = new Block();
|
2019-02-02 08:42:18 +08:00
|
|
|
|
|
|
|
// Add arguments to the entry block.
|
|
|
|
for (auto &placeholderArgPair : entryArguments)
|
|
|
|
if (addDefinition(placeholderArgPair.first,
|
|
|
|
block->addArgument(placeholderArgPair.second))) {
|
|
|
|
delete block;
|
|
|
|
return ParseFailure;
|
|
|
|
}
|
|
|
|
|
2019-01-26 04:48:25 +08:00
|
|
|
if (parseBlock(block)) {
|
2019-02-02 08:42:18 +08:00
|
|
|
delete block;
|
2019-01-26 04:48:25 +08:00
|
|
|
return ParseFailure;
|
|
|
|
}
|
2019-02-02 08:42:18 +08:00
|
|
|
|
|
|
|
// Verify that no other arguments were parsed.
|
|
|
|
if (!entryArguments.empty() &&
|
|
|
|
block->getNumArguments() > entryArguments.size()) {
|
|
|
|
delete block;
|
|
|
|
return emitError("entry block arguments were already defined");
|
|
|
|
}
|
2019-01-26 04:48:25 +08:00
|
|
|
|
|
|
|
// Parse the rest of the block list.
|
2019-02-02 08:42:18 +08:00
|
|
|
results.push_back(block);
|
2019-01-26 04:48:25 +08:00
|
|
|
if (parseBlockListBody(results))
|
|
|
|
return ParseFailure;
|
|
|
|
|
|
|
|
// Reset insertion point to the current block.
|
|
|
|
builder.setInsertionPointToEnd(currentBlock);
|
|
|
|
return ParseSuccess;
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Block list.
|
|
|
|
///
|
|
|
|
/// block-list-body ::= block* '}'
|
|
|
|
///
|
|
|
|
ParseResult
|
|
|
|
FunctionParser::parseBlockListBody(SmallVectorImpl<Block *> &results) {
|
|
|
|
// Parse the block list.
|
|
|
|
while (!consumeIf(Token::r_brace)) {
|
|
|
|
Block *newBlock = nullptr;
|
|
|
|
if (parseBlock(newBlock)) {
|
|
|
|
cleanupInvalidBlocks(results);
|
|
|
|
return ParseFailure;
|
|
|
|
}
|
|
|
|
results.push_back(newBlock);
|
|
|
|
}
|
|
|
|
return ParseSuccess;
|
|
|
|
}
|
|
|
|
|
2018-12-29 10:41:31 +08:00
|
|
|
/// Block declaration.
|
|
|
|
///
|
|
|
|
/// block ::= block-label? instruction* terminator-inst
|
|
|
|
/// block-label ::= block-id block-arg-list? `:`
|
2018-12-30 03:32:37 +08:00
|
|
|
/// block-id ::= caret-id
|
2018-12-29 10:41:31 +08:00
|
|
|
/// block-arg-list ::= `(` ssa-id-and-type-list? `)`
|
|
|
|
///
|
2019-01-26 04:48:25 +08:00
|
|
|
ParseResult FunctionParser::parseBlock(Block *&block) {
|
2018-12-30 05:36:59 +08:00
|
|
|
// The first block for a function is already created.
|
|
|
|
if (block) {
|
|
|
|
// The name for a first block is optional.
|
|
|
|
if (getToken().isNot(Token::caret_identifier))
|
|
|
|
return parseBlockBody(block);
|
|
|
|
}
|
|
|
|
|
2018-12-29 10:41:31 +08:00
|
|
|
SMLoc nameLoc = getToken().getLoc();
|
|
|
|
auto name = getTokenSpelling();
|
2018-12-30 03:32:37 +08:00
|
|
|
if (parseToken(Token::caret_identifier, "expected block name"))
|
2018-12-29 10:41:31 +08:00
|
|
|
return ParseFailure;
|
|
|
|
|
2018-12-30 05:36:59 +08:00
|
|
|
block = defineBlockNamed(name, nameLoc, block);
|
2018-12-29 10:41:31 +08:00
|
|
|
|
|
|
|
// Fail if redefinition.
|
|
|
|
if (!block)
|
|
|
|
return emitError(nameLoc, "redefinition of block '" + name.str() + "'");
|
|
|
|
|
|
|
|
// If an argument list is present, parse it.
|
|
|
|
if (consumeIf(Token::l_paren)) {
|
|
|
|
SmallVector<BlockArgument *, 8> bbArgs;
|
|
|
|
if (parseOptionalBlockArgList(bbArgs, block) ||
|
|
|
|
parseToken(Token::r_paren, "expected ')' to end argument list"))
|
|
|
|
return ParseFailure;
|
|
|
|
}
|
|
|
|
|
2018-12-30 01:01:59 +08:00
|
|
|
if (parseToken(Token::colon, "expected ':' after block name"))
|
2018-12-29 10:41:31 +08:00
|
|
|
return ParseFailure;
|
|
|
|
|
|
|
|
return parseBlockBody(block);
|
|
|
|
}
|
|
|
|
|
|
|
|
ParseResult FunctionParser::parseBlockBody(Block *block) {
|
|
|
|
|
|
|
|
// Set the insertion point to the block we want to insert new operations
|
|
|
|
// into.
|
|
|
|
builder.setInsertionPointToEnd(block);
|
|
|
|
|
|
|
|
// Parse the list of operations that make up the body of the block.
|
2018-12-30 05:36:59 +08:00
|
|
|
while (getToken().isNot(Token::caret_identifier, Token::r_brace)) {
|
2018-12-29 10:41:31 +08:00
|
|
|
switch (getToken().getKind()) {
|
|
|
|
default:
|
|
|
|
if (parseOperation())
|
|
|
|
return ParseFailure;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return ParseSuccess;
|
|
|
|
}
|
|
|
|
|
2018-07-21 09:41:34 +08:00
|
|
|
/// Create and remember a new placeholder for a forward reference.
|
2018-12-28 06:35:10 +08:00
|
|
|
Value *FunctionParser::createForwardReferencePlaceholder(SMLoc loc, Type type) {
|
2018-07-21 09:41:34 +08:00
|
|
|
// Forward references are always created as instructions, even in ML
|
|
|
|
// functions, because we just need something with a def/use chain.
|
|
|
|
//
|
2018-10-19 04:54:44 +08:00
|
|
|
// We create these placeholders as having an empty name, which we know
|
|
|
|
// cannot be created through normal user input, allowing us to distinguish
|
|
|
|
// them.
|
2018-10-10 13:08:52 +08:00
|
|
|
auto name = OperationName("placeholder", getContext());
|
2019-02-05 02:34:20 +08:00
|
|
|
auto *inst = Instruction::create(
|
2019-01-26 04:48:25 +08:00
|
|
|
getEncodedSourceLocation(loc), name, /*operands=*/{}, type,
|
2019-02-01 13:25:17 +08:00
|
|
|
/*attributes=*/{}, /*successors=*/{}, /*numBlockLists=*/0,
|
2019-02-03 14:46:43 +08:00
|
|
|
/*resizableOperandList=*/false, 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.
|
2018-12-28 06:35:10 +08:00
|
|
|
Value *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);
|
|
|
|
|
2019-01-02 23:00:07 +08:00
|
|
|
// Otherwise, this is a forward reference. Create a placeholder and remember
|
2018-08-01 14:14:16 +08:00
|
|
|
// that we did so.
|
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
|
|
|
}
|
|
|
|
|
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-12-29 10:41:31 +08:00
|
|
|
ParseResult FunctionParser::finalizeFunction(SMLoc loc) {
|
2018-10-19 04:54:44 +08:00
|
|
|
// Check for any forward references that are left. If we find any, error
|
|
|
|
// out.
|
2018-07-21 09:41:34 +08:00
|
|
|
if (!forwardReferencePlaceholders.empty()) {
|
2018-12-28 06:35:10 +08:00
|
|
|
SmallVector<std::pair<const char *, Value *>, 4> errors;
|
2018-10-14 22:55:29 +08:00
|
|
|
// Iteration over the map isn't deterministic, so sort by source location.
|
2018-07-21 09:41:34 +08:00
|
|
|
for (auto entry : forwardReferencePlaceholders)
|
|
|
|
errors.push_back({entry.second.getPointer(), entry.first});
|
|
|
|
llvm::array_pod_sort(errors.begin(), errors.end());
|
|
|
|
|
2018-10-14 22:55:29 +08:00
|
|
|
for (auto entry : errors) {
|
|
|
|
auto loc = SMLoc::getFromPointer(entry.first);
|
|
|
|
emitError(loc, "use of undeclared SSA value name");
|
|
|
|
}
|
2018-07-21 09:41:34 +08:00
|
|
|
return ParseFailure;
|
|
|
|
}
|
|
|
|
|
|
|
|
return ParseSuccess;
|
2018-07-19 23:35:28 +08:00
|
|
|
}
|
|
|
|
|
2018-10-14 22:55:29 +08:00
|
|
|
FunctionParser::~FunctionParser() {
|
|
|
|
for (auto &fwd : forwardReferencePlaceholders) {
|
|
|
|
// Drop all uses of undefined forward declared reference and destroy
|
2019-01-08 00:16:49 +08:00
|
|
|
// defining instruction.
|
|
|
|
fwd.first->dropAllUses();
|
2018-10-14 22:55:29 +08:00
|
|
|
fwd.first->getDefiningInst()->destroy();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-12-29 10:41:31 +08:00
|
|
|
/// Register a definition of a value with the symbol table.
|
|
|
|
ParseResult FunctionParser::addDefinition(SSAUseInfo useInfo, Value *value) {
|
|
|
|
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");
|
|
|
|
}
|
|
|
|
|
|
|
|
// 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);
|
|
|
|
}
|
|
|
|
|
|
|
|
entries[useInfo.number].first = value;
|
|
|
|
entries[useInfo.number].second = useInfo.loc;
|
|
|
|
return ParseSuccess;
|
|
|
|
}
|
|
|
|
|
2018-12-29 08:05:35 +08:00
|
|
|
/// Parse a SSA operand for an instruction or instruction.
|
2018-07-08 06:48:26 +08:00
|
|
|
///
|
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(
|
2018-10-31 05:59:22 +08:00
|
|
|
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-10-31 05:59:22 +08:00
|
|
|
auto type = parseType();
|
2018-07-19 23:35:28 +08:00
|
|
|
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
|
2018-11-13 06:58:53 +08:00
|
|
|
/// followed by a type list.
|
2018-07-08 06:48:26 +08:00
|
|
|
///
|
2018-11-13 06:58:53 +08:00
|
|
|
/// ssa-use-and-type-list
|
2018-07-24 02:56:17 +08:00
|
|
|
/// ::= 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-11-13 06:58:53 +08:00
|
|
|
SmallVectorImpl<ValueTy *> &results) {
|
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 there were no operands, then there is no colon or type lists.
|
|
|
|
if (valueIDs.empty())
|
|
|
|
return ParseSuccess;
|
|
|
|
|
2018-10-31 05:59:22 +08:00
|
|
|
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-12-30 01:01:59 +08:00
|
|
|
/// Get the block with the specified name, creating it if it doesn't already
|
|
|
|
/// exist. The location specified is the point of use, which allows
|
2018-12-29 10:41:31 +08:00
|
|
|
/// us to diagnose references to blocks that are not defined precisely.
|
|
|
|
Block *FunctionParser::getBlockNamed(StringRef name, SMLoc loc) {
|
|
|
|
auto &blockAndLoc = blocksByName[name];
|
|
|
|
if (!blockAndLoc.first) {
|
|
|
|
blockAndLoc.first = new Block();
|
|
|
|
forwardRef[blockAndLoc.first] = loc;
|
|
|
|
blockAndLoc.second = loc;
|
|
|
|
}
|
|
|
|
|
|
|
|
return blockAndLoc.first;
|
|
|
|
}
|
|
|
|
|
2018-12-30 01:01:59 +08:00
|
|
|
/// Define the block with the specified name. Returns the Block* or nullptr in
|
|
|
|
/// the case of redefinition.
|
2018-12-30 05:36:59 +08:00
|
|
|
Block *FunctionParser::defineBlockNamed(StringRef name, SMLoc loc,
|
|
|
|
Block *existing) {
|
2018-12-29 10:41:31 +08:00
|
|
|
auto &blockAndLoc = blocksByName[name];
|
|
|
|
if (!blockAndLoc.first) {
|
2018-12-30 05:36:59 +08:00
|
|
|
// If the caller provided a block, use it. Otherwise create a new one.
|
|
|
|
if (!existing)
|
2019-01-26 04:48:25 +08:00
|
|
|
existing = new Block();
|
2018-12-30 05:36:59 +08:00
|
|
|
blockAndLoc.first = existing;
|
2018-12-29 10:41:31 +08:00
|
|
|
blockAndLoc.second = loc;
|
|
|
|
return blockAndLoc.first;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Forward declarations are removed once defined, so if we are defining a
|
|
|
|
// existing block and it is not a forward declaration, then it is a
|
|
|
|
// redeclaration.
|
|
|
|
if (!forwardRef.erase(blockAndLoc.first))
|
|
|
|
return nullptr;
|
|
|
|
return blockAndLoc.first;
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Parse a single operation successor and it's operand list.
|
|
|
|
///
|
|
|
|
/// successor ::= block-id branch-use-list?
|
|
|
|
/// branch-use-list ::= `(` ssa-use-list ':' type-list-no-parens `)`
|
|
|
|
///
|
|
|
|
bool FunctionParser::parseSuccessorAndUseList(
|
|
|
|
Block *&dest, SmallVectorImpl<Value *> &operands) {
|
|
|
|
// Verify branch is identifier and get the matching block.
|
2018-12-30 03:32:37 +08:00
|
|
|
if (!getToken().is(Token::caret_identifier))
|
2018-12-30 01:01:59 +08:00
|
|
|
return emitError("expected block name");
|
2018-12-29 10:41:31 +08:00
|
|
|
dest = getBlockNamed(getTokenSpelling(), getToken().getLoc());
|
|
|
|
consumeToken();
|
|
|
|
|
|
|
|
// Handle optional arguments.
|
|
|
|
if (consumeIf(Token::l_paren) &&
|
|
|
|
(parseOptionalSSAUseAndTypeList(operands) ||
|
|
|
|
parseToken(Token::r_paren, "expected ')' to close argument list"))) {
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2019-01-10 04:28:30 +08:00
|
|
|
/// Parse a comma-separated list of operation successors in brackets.
|
|
|
|
///
|
|
|
|
/// successor-list ::= `[` successor (`,` successor )* `]`
|
|
|
|
///
|
|
|
|
ParseResult FunctionParser::parseSuccessors(
|
|
|
|
SmallVectorImpl<Block *> &destinations,
|
|
|
|
SmallVectorImpl<SmallVector<Value *, 4>> &operands) {
|
|
|
|
if (parseToken(Token::l_square, "expected '['"))
|
|
|
|
return ParseFailure;
|
|
|
|
|
|
|
|
auto parseElt = [this, &destinations, &operands]() {
|
|
|
|
Block *dest;
|
|
|
|
SmallVector<Value *, 4> destOperands;
|
|
|
|
bool r = parseSuccessorAndUseList(dest, destOperands);
|
|
|
|
destinations.push_back(dest);
|
|
|
|
operands.push_back(destOperands);
|
|
|
|
return r ? ParseFailure : ParseSuccess;
|
|
|
|
};
|
|
|
|
return parseCommaSeparatedListUntil(Token::r_square, parseElt,
|
|
|
|
/*allowEmptyList=*/false);
|
|
|
|
}
|
|
|
|
|
2018-12-30 01:01:59 +08:00
|
|
|
/// Parse a (possibly empty) list of SSA operands with types as block arguments.
|
2018-12-29 10:41:31 +08:00
|
|
|
///
|
|
|
|
/// ssa-id-and-type-list ::= ssa-id-and-type (`,` ssa-id-and-type)*
|
|
|
|
///
|
|
|
|
ParseResult FunctionParser::parseOptionalBlockArgList(
|
|
|
|
SmallVectorImpl<BlockArgument *> &results, Block *owner) {
|
|
|
|
if (getToken().is(Token::r_brace))
|
|
|
|
return ParseSuccess;
|
|
|
|
|
2018-12-30 05:36:59 +08:00
|
|
|
// If the block already has arguments, then we're handling the entry block.
|
|
|
|
// Parse and register the names for the arguments, but do not add them.
|
|
|
|
bool definingExistingArgs = owner->getNumArguments() != 0;
|
|
|
|
unsigned nextArgument = 0;
|
|
|
|
|
2018-12-29 10:41:31 +08:00
|
|
|
return parseCommaSeparatedList([&]() -> ParseResult {
|
|
|
|
auto type = parseSSADefOrUseAndType<Type>(
|
|
|
|
[&](SSAUseInfo useInfo, Type type) -> Type {
|
2018-12-30 05:36:59 +08:00
|
|
|
BlockArgument *arg;
|
|
|
|
if (!definingExistingArgs) {
|
|
|
|
arg = owner->addArgument(type);
|
|
|
|
} else if (nextArgument >= owner->getNumArguments()) {
|
|
|
|
emitError("too many arguments specified in argument list");
|
|
|
|
return {};
|
|
|
|
} else {
|
|
|
|
arg = owner->getArgument(nextArgument++);
|
|
|
|
if (arg->getType() != type) {
|
|
|
|
emitError("argument and block argument type mismatch");
|
|
|
|
return {};
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-12-29 10:41:31 +08:00
|
|
|
if (addDefinition(useInfo, arg))
|
|
|
|
return {};
|
|
|
|
return type;
|
|
|
|
});
|
|
|
|
return type ? ParseSuccess : ParseFailure;
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Parse an operation.
|
2018-07-17 02:47:09 +08:00
|
|
|
///
|
|
|
|
/// operation ::=
|
|
|
|
/// (ssa-id `=`)? string '(' ssa-use-list? ')' attribute-dict?
|
2019-01-24 05:11:23 +08:00
|
|
|
/// `:` function-type trailing-location?
|
2018-07-17 02:47:09 +08:00
|
|
|
///
|
2018-12-29 10:41:31 +08:00
|
|
|
ParseResult FunctionParser::parseOperation() {
|
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
|
|
|
}
|
|
|
|
|
2019-02-05 02:34:20 +08:00
|
|
|
Instruction *op;
|
2018-07-26 02:15:20 +08:00
|
|
|
if (getToken().is(Token::bare_identifier) || getToken().isKeyword())
|
2018-12-29 10:41:31 +08:00
|
|
|
op = parseCustomOperation();
|
2018-07-26 02:15:20 +08:00
|
|
|
else if (getToken().is(Token::string))
|
2019-01-24 03:26:56 +08:00
|
|
|
op = parseGenericOperation();
|
2018-07-26 02:15:20 +08:00
|
|
|
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;
|
|
|
|
|
|
|
|
// 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
|
|
|
}
|
|
|
|
|
2019-01-24 05:11:23 +08:00
|
|
|
// Try to parse the optional trailing location.
|
|
|
|
if (parseOptionalTrailingLocation(op))
|
|
|
|
return ParseFailure;
|
|
|
|
|
2018-07-26 02:15:20 +08:00
|
|
|
return ParseSuccess;
|
|
|
|
}
|
|
|
|
|
2019-02-05 02:34:20 +08:00
|
|
|
Instruction *FunctionParser::parseGenericOperation() {
|
2018-08-24 05:32:25 +08:00
|
|
|
// Get location information for the operation.
|
2018-11-09 04:28:35 +08:00
|
|
|
auto srcLocation = getEncodedSourceLocation(getToken().getLoc());
|
2018-08-24 05:32:25 +08:00
|
|
|
|
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-10-15 22:09:18 +08:00
|
|
|
if (name.find('\0') != StringRef::npos)
|
|
|
|
return (emitError("null character not allowed in operation name"), 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
|
|
|
|
2019-02-01 13:25:17 +08:00
|
|
|
// Generic operations have a resizable operation list.
|
|
|
|
result.setOperandListToResizable();
|
|
|
|
|
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
|
|
|
|
2019-01-10 04:28:30 +08:00
|
|
|
// Parse the successor list but don't add successors to the result yet to
|
|
|
|
// avoid messing up with the argument order.
|
|
|
|
SmallVector<Block *, 2> successors;
|
|
|
|
SmallVector<SmallVector<Value *, 4>, 2> successorOperands;
|
|
|
|
if (getToken().is(Token::l_square)) {
|
|
|
|
// Check if the operation is a known terminator.
|
|
|
|
const AbstractOperation *abstractOp = result.name.getAbstractOperation();
|
2019-02-09 01:52:26 +08:00
|
|
|
if (abstractOp && !abstractOp->hasProperty(OperationProperty::Terminator))
|
2019-01-10 04:28:30 +08:00
|
|
|
return emitError("successors in non-terminator"), nullptr;
|
|
|
|
if (parseSuccessors(successors, successorOperands))
|
|
|
|
return nullptr;
|
|
|
|
}
|
|
|
|
|
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-10-31 05:59:22 +08:00
|
|
|
auto fnType = type.dyn_cast<FunctionType>();
|
2018-07-19 06:31:25 +08:00
|
|
|
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-10-31 05:59:22 +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.
|
2018-10-31 05:59:22 +08:00
|
|
|
auto operandTypes = fnType.getInputs();
|
2018-07-19 23:35:28 +08:00
|
|
|
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
|
|
|
|
2019-01-10 04:28:30 +08:00
|
|
|
// Add the sucessors, and their operands after the proper operands.
|
|
|
|
for (const auto &succ : llvm::zip(successors, successorOperands)) {
|
|
|
|
Block *successor = std::get<0>(succ);
|
|
|
|
const SmallVector<Value *, 4> &operands = std::get<1>(succ);
|
|
|
|
result.addSuccessor(successor, operands);
|
|
|
|
}
|
|
|
|
|
2019-01-26 04:48:25 +08:00
|
|
|
// Parse the optional block lists for this operation.
|
|
|
|
std::vector<SmallVector<Block *, 2>> blocks;
|
|
|
|
while (getToken().is(Token::l_brace)) {
|
|
|
|
SmallVector<Block *, 2> newBlocks;
|
2019-02-02 08:42:18 +08:00
|
|
|
if (parseOperationBlockList(newBlocks, /*entryArguments=*/llvm::None)) {
|
2019-01-26 04:48:25 +08:00
|
|
|
for (auto &blockList : blocks)
|
|
|
|
cleanupInvalidBlocks(blockList);
|
|
|
|
return nullptr;
|
|
|
|
}
|
|
|
|
blocks.emplace_back(newBlocks);
|
|
|
|
}
|
|
|
|
result.reserveBlockLists(blocks.size());
|
|
|
|
|
|
|
|
auto *opInst = builder.createOperation(result);
|
|
|
|
|
|
|
|
// Initialize the parsed block lists.
|
|
|
|
for (unsigned i = 0, e = blocks.size(); i != e; ++i) {
|
|
|
|
auto &blockList = opInst->getBlockList(i).getBlocks();
|
|
|
|
blockList.insert(blockList.end(), blocks[i].begin(), blocks[i].end());
|
|
|
|
}
|
|
|
|
return opInst;
|
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) {}
|
|
|
|
|
2019-02-02 08:42:18 +08:00
|
|
|
bool parseOperation(const AbstractOperation *opDefinition,
|
|
|
|
OperationState *opState) {
|
|
|
|
if (opDefinition->parseAssembly(this, opState))
|
|
|
|
return true;
|
|
|
|
|
|
|
|
// Check that enough block lists were reserved for those that were parsed.
|
|
|
|
if (parsedBlockLists.size() > opState->numBlockLists) {
|
|
|
|
return emitError(
|
|
|
|
nameLoc,
|
|
|
|
"parsed more block lists than those reserved in the operation state");
|
|
|
|
}
|
|
|
|
|
|
|
|
// Check there were no dangling entry block arguments.
|
|
|
|
if (!parsedBlockListEntryArguments.empty()) {
|
|
|
|
return emitError(
|
|
|
|
nameLoc,
|
|
|
|
"no block list was attached to parsed entry block arguments");
|
|
|
|
}
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2018-07-26 02:15:20 +08:00
|
|
|
//===--------------------------------------------------------------------===//
|
|
|
|
// 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 ','");
|
|
|
|
}
|
2019-02-02 08:42:18 +08:00
|
|
|
bool parseEqual() override {
|
|
|
|
return parser.parseToken(Token::equal, "expected '='");
|
|
|
|
}
|
2018-07-26 02:15:20 +08:00
|
|
|
|
2018-11-16 09:53:51 +08:00
|
|
|
bool parseType(Type &result) override {
|
|
|
|
return !(result = parser.parseType());
|
|
|
|
}
|
|
|
|
|
2018-10-31 05:59:22 +08:00
|
|
|
bool parseColonType(Type &result) override {
|
2018-08-24 05:58:27 +08:00
|
|
|
return parser.parseToken(Token::colon, "expected ':'") ||
|
|
|
|
!(result = parser.parseType());
|
2018-07-26 02:15:20 +08:00
|
|
|
}
|
|
|
|
|
2018-10-31 05:59:22 +08:00
|
|
|
bool parseColonTypeList(SmallVectorImpl<Type> &result) override {
|
2018-08-24 05:58:27 +08:00
|
|
|
if (parser.parseToken(Token::colon, "expected ':'"))
|
2018-07-26 02:15:20 +08:00
|
|
|
return true;
|
|
|
|
|
|
|
|
do {
|
2018-10-31 05:59:22 +08:00
|
|
|
if (auto type = parser.parseType())
|
2018-07-26 02:15:20 +08:00
|
|
|
result.push_back(type);
|
|
|
|
else
|
|
|
|
return true;
|
|
|
|
|
|
|
|
} while (parser.consumeIf(Token::comma));
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2018-12-06 07:30:25 +08:00
|
|
|
bool parseTrailingOperandList(SmallVectorImpl<OperandType> &result,
|
|
|
|
int requiredOperandCount,
|
|
|
|
Delimiter delimiter) override {
|
|
|
|
if (parser.getToken().is(Token::comma)) {
|
|
|
|
parseComma();
|
|
|
|
return parseOperandList(result, requiredOperandCount, delimiter);
|
|
|
|
}
|
|
|
|
if (requiredOperandCount != -1)
|
|
|
|
return emitError(parser.getToken().getLoc(),
|
|
|
|
"expected " + Twine(requiredOperandCount) + " operands");
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2019-01-29 13:23:53 +08:00
|
|
|
/// Parse an optional keyword.
|
|
|
|
bool parseOptionalKeyword(const char *keyword) override {
|
|
|
|
// Check that the current token is a bare identifier or keyword.
|
|
|
|
if (parser.getToken().isNot(Token::bare_identifier) &&
|
|
|
|
!parser.getToken().isKeyword())
|
|
|
|
return true;
|
|
|
|
|
|
|
|
if (parser.getTokenSpelling() == keyword) {
|
|
|
|
parser.consumeToken();
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
return true;
|
2018-09-14 00:16:32 +08:00
|
|
|
}
|
|
|
|
|
2018-11-16 09:53:51 +08:00
|
|
|
/// Parse an arbitrary attribute of a given type and return it in result. This
|
|
|
|
/// also adds the attribute to the specified attribute list with the specified
|
|
|
|
/// name.
|
|
|
|
bool parseAttribute(Attribute &result, Type type, const char *attrName,
|
2018-08-24 05:58:27 +08:00
|
|
|
SmallVectorImpl<NamedAttribute> &attrs) override {
|
2018-11-16 09:53:51 +08:00
|
|
|
result = parser.parseAttribute(type);
|
2018-08-03 07:54:36 +08:00
|
|
|
if (!result)
|
|
|
|
return true;
|
|
|
|
|
|
|
|
attrs.push_back(
|
|
|
|
NamedAttribute(parser.builder.getIdentifier(attrName), result));
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2018-11-16 09:53:51 +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.
|
|
|
|
bool parseAttribute(Attribute &result, const char *attrName,
|
|
|
|
SmallVectorImpl<NamedAttribute> &attrs) override {
|
|
|
|
return parseAttribute(result, Type(), attrName, attrs);
|
|
|
|
}
|
|
|
|
|
2018-08-03 07:54:36 +08:00
|
|
|
/// 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;
|
|
|
|
}
|
|
|
|
|
2018-12-29 05:07:39 +08:00
|
|
|
bool parseSuccessorAndUseList(Block *&dest,
|
2018-12-28 06:35:10 +08:00
|
|
|
SmallVectorImpl<Value *> &operands) override {
|
2018-11-16 01:56:06 +08:00
|
|
|
// Defer successor parsing to the function parsers.
|
|
|
|
return parser.parseSuccessorAndUseList(dest, operands);
|
|
|
|
}
|
|
|
|
|
2018-07-26 02:15:20 +08:00
|
|
|
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-09-10 08:59:22 +08:00
|
|
|
// Don't check for the absence of a delimiter if the number of operands
|
|
|
|
// is unknown (and hence the operand list could be empty).
|
|
|
|
if (requiredOperandCount == -1)
|
|
|
|
break;
|
|
|
|
// Token already matches an identifier and so can't be a delimiter.
|
|
|
|
if (parser.getToken().is(Token::percent_identifier))
|
|
|
|
break;
|
|
|
|
// Test against known delimiters.
|
|
|
|
if (parser.getToken().is(Token::l_paren) ||
|
|
|
|
parser.getToken().is(Token::l_square))
|
|
|
|
return emitError(startLoc, "unexpected delimiter");
|
2018-09-10 23:26:58 +08:00
|
|
|
return emitError(startLoc, "invalid operand");
|
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)
|
2018-09-10 04:41:19 +08:00
|
|
|
return emitError(startLoc,
|
|
|
|
"expected " + Twine(requiredOperandCount) + " operands");
|
2018-07-26 02:15:20 +08:00
|
|
|
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.
|
2018-10-31 05:59:22 +08:00
|
|
|
virtual bool resolveFunctionName(StringRef name, FunctionType type,
|
2018-08-22 08:55:22 +08:00
|
|
|
llvm::SMLoc loc, Function *&result) {
|
|
|
|
result = parser.resolveFunctionReference(name, loc, type);
|
|
|
|
return result == nullptr;
|
|
|
|
}
|
|
|
|
|
2019-01-29 13:23:53 +08:00
|
|
|
/// Parses a list of blocks.
|
|
|
|
bool parseBlockList() override {
|
2019-02-02 08:42:18 +08:00
|
|
|
// Parse the block list.
|
2019-01-29 13:23:53 +08:00
|
|
|
SmallVector<Block *, 2> results;
|
2019-02-02 08:42:18 +08:00
|
|
|
if (parser.parseOperationBlockList(results, parsedBlockListEntryArguments))
|
2019-01-29 13:23:53 +08:00
|
|
|
return true;
|
2019-02-02 08:42:18 +08:00
|
|
|
|
|
|
|
parsedBlockListEntryArguments.clear();
|
2019-01-29 13:23:53 +08:00
|
|
|
parsedBlockLists.emplace_back(results);
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2019-02-02 08:42:18 +08:00
|
|
|
/// Parses an argument for the entry block of the next block list to be
|
|
|
|
/// parsed.
|
|
|
|
bool parseBlockListEntryBlockArgument(Type argType) override {
|
|
|
|
SmallVector<Value *, 1> argValues;
|
|
|
|
OperandType operand;
|
|
|
|
if (parseOperand(operand))
|
|
|
|
return true;
|
|
|
|
|
|
|
|
// Create a place holder for this argument.
|
|
|
|
FunctionParser::SSAUseInfo operandInfo = {operand.name, operand.number,
|
|
|
|
operand.location};
|
|
|
|
if (auto *value = parser.resolveSSAUse(operandInfo, argType)) {
|
|
|
|
parsedBlockListEntryArguments.emplace_back(operandInfo, argType);
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
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-10-31 05:59:22 +08:00
|
|
|
bool resolveOperand(const OperandType &operand, Type type,
|
2018-12-28 06:35:10 +08:00
|
|
|
SmallVectorImpl<Value *> &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 {
|
2019-01-29 13:23:53 +08:00
|
|
|
// If we emit an error, then cleanup any parsed block lists.
|
|
|
|
for (auto &blockList : parsedBlockLists)
|
|
|
|
parser.cleanupInvalidBlocks(blockList);
|
|
|
|
parsedBlockLists.clear();
|
|
|
|
|
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; }
|
|
|
|
|
2019-01-29 13:23:53 +08:00
|
|
|
/// Returns the block lists that were parsed.
|
|
|
|
MutableArrayRef<SmallVector<Block *, 2>> getParsedBlockLists() {
|
|
|
|
return parsedBlockLists;
|
|
|
|
}
|
|
|
|
|
2018-07-26 02:15:20 +08:00
|
|
|
private:
|
2019-01-29 13:23:53 +08:00
|
|
|
std::vector<SmallVector<Block *, 2>> parsedBlockLists;
|
2019-02-02 08:42:18 +08:00
|
|
|
SmallVector<std::pair<FunctionParser::SSAUseInfo, Type>, 2>
|
|
|
|
parsedBlockListEntryArguments;
|
2018-07-26 02:15:20 +08:00
|
|
|
SMLoc nameLoc;
|
|
|
|
StringRef opName;
|
|
|
|
FunctionParser &parser;
|
|
|
|
bool emittedError = false;
|
|
|
|
};
|
|
|
|
} // end anonymous namespace.
|
|
|
|
|
2019-02-05 02:34:20 +08:00
|
|
|
Instruction *FunctionParser::parseCustomOperation() {
|
2018-07-26 02:15:20 +08:00
|
|
|
auto opLoc = getToken().getLoc();
|
|
|
|
auto opName = getTokenSpelling();
|
|
|
|
CustomOpAsmParser opAsmParser(opLoc, opName, *this);
|
|
|
|
|
2018-10-22 10:49:31 +08:00
|
|
|
auto *opDefinition = AbstractOperation::lookup(opName, getContext());
|
2018-07-26 02:15:20 +08:00
|
|
|
if (!opDefinition) {
|
|
|
|
opAsmParser.emitError(opLoc, "is unknown");
|
|
|
|
return nullptr;
|
|
|
|
}
|
|
|
|
|
|
|
|
consumeToken();
|
|
|
|
|
2018-10-19 04:54:44 +08:00
|
|
|
// If the custom op parser crashes, produce some indication to help
|
|
|
|
// debugging.
|
2018-08-22 08:55:22 +08:00
|
|
|
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.
|
2018-11-09 04:28:35 +08:00
|
|
|
auto srcLocation = getEncodedSourceLocation(opLoc);
|
2018-08-24 05:32:25 +08:00
|
|
|
|
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);
|
2019-02-02 08:42:18 +08:00
|
|
|
if (opAsmParser.parseOperation(opDefinition, &opState))
|
2018-08-08 00:12:35 +08:00
|
|
|
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.
|
2019-01-29 13:23:53 +08:00
|
|
|
auto *opInst = builder.createOperation(opState);
|
|
|
|
|
|
|
|
// Resolve any parsed block lists.
|
2019-02-02 08:42:18 +08:00
|
|
|
auto parsedBlockLists = opAsmParser.getParsedBlockLists();
|
2019-01-29 13:23:53 +08:00
|
|
|
for (unsigned i = 0, e = parsedBlockLists.size(); i != e; ++i) {
|
|
|
|
auto &opBlockList = opInst->getBlockList(i).getBlocks();
|
|
|
|
opBlockList.insert(opBlockList.end(), parsedBlockLists[i].begin(),
|
|
|
|
parsedBlockLists[i].end());
|
|
|
|
}
|
|
|
|
return opInst;
|
2018-06-29 08:02:32 +08:00
|
|
|
}
|
|
|
|
|
2018-08-08 05:24:38 +08:00
|
|
|
/// Parse an affine constraint.
|
|
|
|
/// affine-constraint ::= affine-expr `>=` `0`
|
|
|
|
/// | affine-expr `==` `0`
|
|
|
|
///
|
2018-10-19 04:54:44 +08:00
|
|
|
/// isEq is set to true if the parsed constraint is an equality, false if it
|
|
|
|
/// is an inequality (greater than or equal).
|
2018-08-08 05:24:38 +08:00
|
|
|
///
|
2018-10-09 04:47:18 +08:00
|
|
|
AffineExpr AffineParser::parseAffineConstraint(bool *isEq) {
|
|
|
|
AffineExpr expr = parseAffineExpr();
|
2018-08-08 05:24:38 +08:00
|
|
|
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);
|
|
|
|
}
|
|
|
|
|
2018-10-26 09:33:42 +08:00
|
|
|
/// Parse the constraints that are part of an integer set definition.
|
2018-08-08 05:24:38 +08:00
|
|
|
/// integer-set-inline
|
2018-10-19 04:54:44 +08:00
|
|
|
/// ::= dim-and-symbol-id-lists `:`
|
2019-02-03 13:01:11 +08:00
|
|
|
/// '(' affine-constraint-conjunction? ')'
|
|
|
|
/// affine-constraint-conjunction ::= affine-constraint (`,`
|
|
|
|
/// affine-constraint)*
|
2018-08-08 05:24:38 +08:00
|
|
|
///
|
2018-10-26 09:33:42 +08:00
|
|
|
IntegerSet AffineParser::parseIntegerSetConstraints(unsigned numDims,
|
|
|
|
unsigned numSymbols) {
|
2019-01-15 01:45:09 +08:00
|
|
|
if (parseToken(Token::l_paren,
|
|
|
|
"expected '(' at start of integer set constraint list"))
|
2019-01-27 02:41:17 +08:00
|
|
|
return IntegerSet();
|
2018-08-08 05:24:38 +08:00
|
|
|
|
2018-10-09 04:47:18 +08:00
|
|
|
SmallVector<AffineExpr, 4> constraints;
|
2018-08-08 05:24:38 +08:00
|
|
|
SmallVector<bool, 4> isEqs;
|
|
|
|
auto parseElt = [&]() -> ParseResult {
|
|
|
|
bool isEq;
|
2018-10-08 23:09:50 +08:00
|
|
|
auto elt = parseAffineConstraint(&isEq);
|
2018-08-08 05:24:38 +08:00
|
|
|
ParseResult res = elt ? ParseSuccess : ParseFailure;
|
|
|
|
if (elt) {
|
|
|
|
constraints.push_back(elt);
|
|
|
|
isEqs.push_back(isEq);
|
|
|
|
}
|
|
|
|
return res;
|
|
|
|
};
|
|
|
|
|
2019-02-03 13:01:11 +08:00
|
|
|
// Parse a list of affine constraints (comma-separated).
|
2018-08-08 05:24:38 +08:00
|
|
|
if (parseCommaSeparatedListUntil(Token::r_paren, parseElt, true))
|
2019-01-27 02:41:17 +08:00
|
|
|
return IntegerSet();
|
2019-01-15 05:56:59 +08:00
|
|
|
|
2019-02-03 13:01:11 +08:00
|
|
|
// If no constraints were parsed, then treat this as a degenerate 'true' case.
|
2019-01-15 05:56:59 +08:00
|
|
|
if (constraints.empty()) {
|
2019-02-03 13:01:11 +08:00
|
|
|
/* 0 == 0 */
|
|
|
|
auto zero = getAffineConstantExpr(0, getContext());
|
|
|
|
return builder.getIntegerSet(numDims, numSymbols, zero, true);
|
2019-01-15 05:56:59 +08:00
|
|
|
}
|
2018-08-08 05:24:38 +08:00
|
|
|
|
|
|
|
// Parsed a valid integer set.
|
|
|
|
return builder.getIntegerSet(numDims, numSymbols, constraints, isEqs);
|
|
|
|
}
|
|
|
|
|
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-10-26 09:33:42 +08:00
|
|
|
ParseResult parseAffineStructureDef();
|
2018-07-11 01:08:27 +08:00
|
|
|
|
2019-01-08 10:42:04 +08:00
|
|
|
ParseResult parseTypeAliasDef();
|
|
|
|
|
2018-07-11 01:08:27 +08:00
|
|
|
// Functions.
|
2018-12-30 01:01:59 +08:00
|
|
|
ParseResult parseArgumentList(SmallVectorImpl<Type> &argTypes,
|
|
|
|
SmallVectorImpl<StringRef> &argNames);
|
2018-10-31 05:59:22 +08:00
|
|
|
ParseResult parseFunctionSignature(StringRef &name, FunctionType &type,
|
2018-12-30 01:01:59 +08:00
|
|
|
SmallVectorImpl<StringRef> &argNames);
|
2019-01-03 02:20:00 +08:00
|
|
|
ParseResult parseFunc();
|
2018-07-11 01:08:27 +08:00
|
|
|
};
|
|
|
|
} // end anonymous namespace
|
|
|
|
|
2018-10-26 09:33:42 +08:00
|
|
|
/// Parses either an affine map declaration or an integer set declaration.
|
|
|
|
///
|
2018-07-11 01:08:27 +08:00
|
|
|
/// Affine map declaration.
|
|
|
|
///
|
|
|
|
/// affine-map-def ::= affine-map-id `=` affine-map-inline
|
|
|
|
///
|
2018-10-26 09:33:42 +08:00
|
|
|
/// Integer set declaration.
|
|
|
|
///
|
|
|
|
/// integer-set-decl ::= integer-set-id `=` integer-set-inline
|
|
|
|
///
|
|
|
|
ParseResult ModuleParser::parseAffineStructureDef() {
|
2018-07-11 01:08:27 +08:00
|
|
|
assert(getToken().is(Token::hash_identifier));
|
|
|
|
|
2018-10-26 09:33:42 +08:00
|
|
|
StringRef affineStructureId = getTokenSpelling().drop_front();
|
2018-07-11 01:08:27 +08:00
|
|
|
|
|
|
|
// Check for redefinitions.
|
2018-10-26 09:33:42 +08:00
|
|
|
if (getState().affineMapDefinitions.count(affineStructureId) > 0)
|
|
|
|
return emitError("redefinition of affine map id '" + affineStructureId +
|
|
|
|
"'");
|
|
|
|
if (getState().integerSetDefinitions.count(affineStructureId) > 0)
|
|
|
|
return emitError("redefinition of integer set id '" + affineStructureId +
|
|
|
|
"'");
|
2018-07-11 01:08:27 +08:00
|
|
|
|
|
|
|
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-10-26 09:33:42 +08:00
|
|
|
AffineMap map;
|
|
|
|
IntegerSet set;
|
2019-01-27 02:41:17 +08:00
|
|
|
if (AffineParser(getState()).parseAffineMapOrIntegerSetInline(map, set))
|
2018-08-08 05:24:38 +08:00
|
|
|
return ParseFailure;
|
|
|
|
|
2019-01-27 02:41:17 +08:00
|
|
|
if (map) {
|
2018-10-26 09:33:42 +08:00
|
|
|
getState().affineMapDefinitions[affineStructureId] = map;
|
2019-01-27 02:41:17 +08:00
|
|
|
return ParseSuccess;
|
|
|
|
}
|
2018-07-11 01:08:27 +08:00
|
|
|
|
2019-01-27 02:41:17 +08:00
|
|
|
assert(set);
|
|
|
|
getState().integerSetDefinitions[affineStructureId] = set;
|
2018-07-11 01:08:27 +08:00
|
|
|
return ParseSuccess;
|
|
|
|
}
|
|
|
|
|
2019-01-08 10:42:04 +08:00
|
|
|
/// Parse a type alias declaration.
|
|
|
|
///
|
|
|
|
/// type-alias-def ::= '!' alias-name `=` 'type' type
|
|
|
|
///
|
|
|
|
ParseResult ModuleParser::parseTypeAliasDef() {
|
|
|
|
assert(getToken().is(Token::exclamation_identifier));
|
|
|
|
|
|
|
|
StringRef aliasName = getTokenSpelling().drop_front();
|
|
|
|
|
|
|
|
// Check for redefinitions.
|
|
|
|
if (getState().typeAliasDefinitions.count(aliasName) > 0)
|
|
|
|
return emitError("redefinition of type alias id '" + aliasName + "'");
|
|
|
|
|
|
|
|
consumeToken(Token::exclamation_identifier);
|
|
|
|
|
|
|
|
// Parse the '=' and 'type'.
|
|
|
|
if (parseToken(Token::equal, "expected '=' in type alias definition") ||
|
|
|
|
parseToken(Token::kw_type, "expected 'type' in type alias definition"))
|
|
|
|
return ParseFailure;
|
|
|
|
|
|
|
|
// Parse the type.
|
|
|
|
Type aliasedType = parseType();
|
|
|
|
if (!aliasedType)
|
|
|
|
return ParseFailure;
|
|
|
|
|
|
|
|
// Register this alias with the parser state.
|
|
|
|
getState().typeAliasDefinitions.try_emplace(aliasName, aliasedType);
|
|
|
|
|
|
|
|
return ParseSuccess;
|
|
|
|
}
|
|
|
|
|
2018-12-29 00:48:09 +08:00
|
|
|
/// Parse a (possibly empty) list of Function arguments with types.
|
2018-07-20 00:52:39 +08:00
|
|
|
///
|
2018-12-30 01:01:59 +08:00
|
|
|
/// named-argument ::= ssa-id `:` type
|
|
|
|
/// argument-list ::= named-argument (`,` named-argument)* | /*empty*/
|
|
|
|
/// argument-list ::= type (`,` type)* | /*empty*/
|
2018-07-20 00:52:39 +08:00
|
|
|
///
|
|
|
|
ParseResult
|
2018-12-30 01:01:59 +08:00
|
|
|
ModuleParser::parseArgumentList(SmallVectorImpl<Type> &argTypes,
|
|
|
|
SmallVectorImpl<StringRef> &argNames) {
|
2018-07-24 08:30:01 +08:00
|
|
|
consumeToken(Token::l_paren);
|
|
|
|
|
2018-12-30 01:01:59 +08:00
|
|
|
// The argument list either has to consistently have ssa-id's followed by
|
|
|
|
// types, or just be a type list. It isn't ok to sometimes have SSA ID's and
|
|
|
|
// sometimes not.
|
2018-07-20 00:52:39 +08:00
|
|
|
auto parseElt = [&]() -> ParseResult {
|
2018-12-30 01:01:59 +08:00
|
|
|
// Parse argument name if present.
|
|
|
|
auto loc = getToken().getLoc();
|
2018-08-07 02:54:39 +08:00
|
|
|
StringRef name = getTokenSpelling();
|
2018-12-30 01:01:59 +08:00
|
|
|
if (consumeIf(Token::percent_identifier)) {
|
|
|
|
// Reject this if the preceding argument was missing a name.
|
|
|
|
if (argNames.empty() && !argTypes.empty())
|
|
|
|
return emitError(loc, "expected type instead of SSA identifier");
|
2018-07-20 00:52:39 +08:00
|
|
|
|
2018-12-30 01:01:59 +08:00
|
|
|
argNames.push_back(name);
|
|
|
|
|
|
|
|
if (parseToken(Token::colon, "expected ':'"))
|
|
|
|
return ParseFailure;
|
|
|
|
} else {
|
|
|
|
// Reject this if the preceding argument had a name.
|
|
|
|
if (!argNames.empty())
|
|
|
|
return emitError("expected SSA identifier");
|
|
|
|
}
|
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-10-19 04:54:44 +08:00
|
|
|
/// Parse a function signature, starting with a name and including the
|
|
|
|
/// parameter list.
|
2018-07-11 01:08:27 +08:00
|
|
|
///
|
2018-12-30 01:01:59 +08:00
|
|
|
/// function-signature ::=
|
|
|
|
/// function-id `(` argument-list `)` (`->` type-list)?
|
2018-07-11 01:08:27 +08:00
|
|
|
///
|
2018-07-20 00:52:39 +08:00
|
|
|
ParseResult
|
2018-10-31 05:59:22 +08:00
|
|
|
ModuleParser::parseFunctionSignature(StringRef &name, FunctionType &type,
|
2018-12-30 01:01:59 +08:00
|
|
|
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-10-31 05:59:22 +08:00
|
|
|
SmallVector<Type, 4> argTypes;
|
2018-12-30 01:01:59 +08:00
|
|
|
if (parseArgumentList(argTypes, argNames))
|
2018-07-11 01:08:27 +08:00
|
|
|
return ParseFailure;
|
|
|
|
|
|
|
|
// Parse the return type if present.
|
2018-10-31 05:59:22 +08:00
|
|
|
SmallVector<Type, 4> results;
|
2018-07-11 01:08:27 +08:00
|
|
|
if (consumeIf(Token::arrow)) {
|
2019-02-06 03:47:02 +08:00
|
|
|
if (parseFunctionResultTypes(results))
|
2018-07-11 01:08:27 +08:00
|
|
|
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;
|
|
|
|
}
|
|
|
|
|
2018-12-30 01:01:59 +08:00
|
|
|
/// Function declarations.
|
2018-09-19 07:36:26 +08:00
|
|
|
///
|
2019-01-24 05:11:23 +08:00
|
|
|
/// function ::= `func` function-signature function-attributes?
|
|
|
|
/// trailing-location? function-body?
|
2019-01-03 04:32:30 +08:00
|
|
|
/// function-body ::= `{` block+ `}`
|
2018-12-30 01:01:59 +08:00
|
|
|
/// function-attributes ::= `attributes` attribute-dict
|
2018-07-11 01:08:27 +08:00
|
|
|
///
|
2019-01-03 02:20:00 +08:00
|
|
|
ParseResult ModuleParser::parseFunc() {
|
2018-12-30 01:01:59 +08:00
|
|
|
consumeToken();
|
2018-07-11 01:08:27 +08:00
|
|
|
|
|
|
|
StringRef name;
|
2018-10-31 05:59:22 +08:00
|
|
|
FunctionType type;
|
2018-12-30 01:01:59 +08:00
|
|
|
SmallVector<StringRef, 4> argNames;
|
2018-07-11 01:08:27 +08:00
|
|
|
|
2018-08-18 07:49:42 +08:00
|
|
|
auto loc = getToken().getLoc();
|
2018-12-30 01:01:59 +08:00
|
|
|
if (parseFunctionSignature(name, type, argNames))
|
2018-07-11 01:08:27 +08:00
|
|
|
return ParseFailure;
|
|
|
|
|
2018-12-30 01:01:59 +08:00
|
|
|
// If function attributes are present, parse them.
|
2018-09-19 07:36:26 +08:00
|
|
|
SmallVector<NamedAttribute, 8> attrs;
|
2018-12-30 01:01:59 +08:00
|
|
|
if (consumeIf(Token::kw_attributes)) {
|
|
|
|
if (parseAttributeDict(attrs))
|
|
|
|
return ParseFailure;
|
2018-09-19 07:36:26 +08:00
|
|
|
}
|
|
|
|
|
2018-12-30 01:01:59 +08:00
|
|
|
// Okay, the function signature was parsed correctly, create the function now.
|
2018-09-19 07:36:26 +08:00
|
|
|
auto *function =
|
2019-01-03 02:20:00 +08:00
|
|
|
new Function(getEncodedSourceLocation(loc), name, type, attrs);
|
2018-08-18 07:49:42 +08:00
|
|
|
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
|
|
|
|
2019-01-24 05:11:23 +08:00
|
|
|
// Parse an optional trailing location.
|
|
|
|
if (parseOptionalTrailingLocation(function))
|
|
|
|
return ParseFailure;
|
|
|
|
|
2018-12-30 01:01:59 +08:00
|
|
|
// External functions have no body.
|
2019-01-03 02:20:00 +08:00
|
|
|
if (getToken().isNot(Token::l_brace))
|
2018-12-30 01:01:59 +08:00
|
|
|
return ParseSuccess;
|
2018-07-11 01:08:27 +08:00
|
|
|
|
2018-12-30 01:01:59 +08:00
|
|
|
// Create the parser.
|
|
|
|
auto parser = FunctionParser(getState(), function);
|
2018-07-20 00:52:39 +08:00
|
|
|
|
2018-12-30 05:36:59 +08:00
|
|
|
bool hadNamedArguments = !argNames.empty();
|
2018-07-11 01:08:27 +08:00
|
|
|
|
2019-01-03 02:20:00 +08:00
|
|
|
// Add the entry block and argument list.
|
|
|
|
function->addEntryBlock();
|
2018-09-19 07:36:26 +08:00
|
|
|
|
2018-08-07 02:54:39 +08:00
|
|
|
// Add definitions of the function arguments.
|
2018-12-30 01:01:59 +08:00
|
|
|
if (hadNamedArguments) {
|
|
|
|
for (unsigned i = 0, e = function->getNumArguments(); i != e; ++i) {
|
|
|
|
if (parser.addDefinition({argNames[i], 0, loc}, function->getArgument(i)))
|
|
|
|
return ParseFailure;
|
|
|
|
}
|
2018-08-07 02:54:39 +08:00
|
|
|
}
|
2018-07-11 01:08:27 +08:00
|
|
|
|
2018-12-30 01:01:59 +08:00
|
|
|
return parser.parseFunctionBody(hadNamedArguments);
|
2018-07-11 01:08:27 +08:00
|
|
|
}
|
|
|
|
|
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.
|
2018-10-26 06:46:10 +08:00
|
|
|
DenseMap<Attribute, 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);
|
2018-09-08 00:08:13 +08:00
|
|
|
if (!resolvedFunction) {
|
|
|
|
forwardRef.second->emitError("reference to undefined function '" +
|
|
|
|
name.str() + "'");
|
|
|
|
return ParseFailure;
|
|
|
|
}
|
2018-08-20 12:17:22 +08:00
|
|
|
|
2018-09-08 00:08:13 +08:00
|
|
|
remappingTable[builder.getFunctionAttr(forwardRef.second)] =
|
2018-08-21 23:42:19 +08:00
|
|
|
builder.getFunctionAttr(resolvedFunction);
|
|
|
|
}
|
|
|
|
|
|
|
|
// If there was nothing to remap, then we're done.
|
|
|
|
if (remappingTable.empty())
|
|
|
|
return ParseSuccess;
|
|
|
|
|
2018-10-19 04:54:44 +08:00
|
|
|
// Otherwise, walk the entire module replacing uses of one attribute set
|
|
|
|
// with the correct ones.
|
2018-11-15 06:26:47 +08:00
|
|
|
remapFunctionAttrs(*getModule(), remappingTable);
|
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)
|
2018-12-28 03:07:34 +08:00
|
|
|
delete forwardRef.second;
|
2018-10-10 05:40:41 +08:00
|
|
|
getState().functionForwardRefs.clear();
|
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
|
2018-10-19 04:54:44 +08:00
|
|
|
// stop. Someday we could introduce error recovery if there was demand
|
|
|
|
// for it.
|
2018-06-23 01:39:19 +08:00
|
|
|
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:
|
2018-10-26 09:33:42 +08:00
|
|
|
if (parseAffineStructureDef())
|
2018-08-08 05:24:38 +08:00
|
|
|
return ParseFailure;
|
|
|
|
break;
|
|
|
|
|
2019-01-08 10:42:04 +08:00
|
|
|
case Token::exclamation_identifier:
|
|
|
|
if (parseTypeAliasDef())
|
|
|
|
return ParseFailure;
|
|
|
|
break;
|
|
|
|
|
2019-01-03 02:20:00 +08:00
|
|
|
case Token::kw_func:
|
|
|
|
if (parseFunc())
|
2018-07-11 01:08:27 +08:00
|
|
|
return ParseFailure;
|
2018-06-29 08:02:32 +08:00
|
|
|
break;
|
2018-06-23 01:39:19 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
|
|
|
|
/// This parses the file specified by the indicated SourceMgr and returns an
|
2018-10-19 04:54:44 +08:00
|
|
|
/// MLIR module if it was valid. If not, it emits diagnostics and returns
|
|
|
|
/// null.
|
2018-09-09 09:37:27 +08:00
|
|
|
Module *mlir::parseSourceFile(const llvm::SourceMgr &sourceMgr,
|
2018-09-03 13:01:45 +08:00
|
|
|
MLIRContext *context) {
|
2018-08-21 23:42:19 +08:00
|
|
|
|
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-09-03 13:01:45 +08:00
|
|
|
ParserState state(sourceMgr, module.get());
|
2018-08-21 23:42:19 +08:00
|
|
|
if (ModuleParser(state).parseModule()) {
|
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
|
|
|
|
2018-10-19 04:54:44 +08:00
|
|
|
// Make sure the parse module has no other structural problems detected by
|
|
|
|
// the verifier.
|
2018-09-08 00:08:13 +08:00
|
|
|
if (module->verify())
|
2018-08-21 23:42:19 +08:00
|
|
|
return nullptr;
|
|
|
|
|
2018-07-11 01:08:27 +08:00
|
|
|
return module.release();
|
2018-06-23 01:39:19 +08:00
|
|
|
}
|
2018-09-03 22:38:31 +08:00
|
|
|
|
2018-10-19 04:54:44 +08:00
|
|
|
/// This parses the program string to a MLIR module if it was valid. If not,
|
|
|
|
/// it emits diagnostics and returns null.
|
2018-09-03 22:38:31 +08:00
|
|
|
Module *mlir::parseSourceString(StringRef moduleStr, MLIRContext *context) {
|
|
|
|
auto memBuffer = MemoryBuffer::getMemBuffer(moduleStr);
|
|
|
|
if (!memBuffer)
|
|
|
|
return nullptr;
|
|
|
|
|
|
|
|
SourceMgr sourceMgr;
|
|
|
|
sourceMgr.AddNewSourceBuffer(std::move(memBuffer), SMLoc());
|
|
|
|
return parseSourceFile(sourceMgr, context);
|
|
|
|
}
|