blob: 21ca18dcfa3001ea5455a37a194701830cc775fc [file] [log] [blame]
/*
* Copyright (C) 2019 Apple Inc. All rights reserved.
*
* Redistribution and use in source and binary forms, with or without
* modification, are permitted provided that the following conditions
* are met:
* 1. Redistributions of source code must retain the above copyright
* notice, this list of conditions and the following disclaimer.
* 2. Redistributions in binary form must reproduce the above copyright
* notice, this list of conditions and the following disclaimer in the
* documentation and/or other materials provided with the distribution.
*
* THIS SOFTWARE IS PROVIDED BY APPLE INC. AND ITS CONTRIBUTORS ``AS IS''
* AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO,
* THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR
* PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL APPLE INC. OR ITS CONTRIBUTORS
* BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR
* CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
* SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
* INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN
* CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
* ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF
* THE POSSIBILITY OF SUCH DAMAGE.
*/
#include "config.h"
#include "WHLSLNameResolver.h"
#if ENABLE(WEBGPU)
#include "WHLSLCallExpression.h"
#include "WHLSLDoWhileLoop.h"
#include "WHLSLDotExpression.h"
#include "WHLSLEnumerationDefinition.h"
#include "WHLSLEnumerationMemberLiteral.h"
#include "WHLSLForLoop.h"
#include "WHLSLFunctionDefinition.h"
#include "WHLSLIfStatement.h"
#include "WHLSLNameContext.h"
#include "WHLSLProgram.h"
#include "WHLSLPropertyAccessExpression.h"
#include "WHLSLResolveOverloadImpl.h"
#include "WHLSLReturn.h"
#include "WHLSLScopedSetAdder.h"
#include "WHLSLTypeReference.h"
#include "WHLSLVariableDeclaration.h"
#include "WHLSLVariableReference.h"
#include "WHLSLWhileLoop.h"
namespace WebCore {
namespace WHLSL {
NameResolver::NameResolver(NameContext& nameContext)
: m_nameContext(nameContext)
{
}
void NameResolver::visit(AST::TypeReference& typeReference)
{
ScopedSetAdder<AST::TypeReference*> adder(m_typeReferences, &typeReference);
if (!adder.isNewEntry()) {
setError();
return;
}
Visitor::visit(typeReference);
if (typeReference.maybeResolvedType()) // FIXME: https://bugs.webkit.org/show_bug.cgi?id=198161 Shouldn't we know by now whether the type has been resolved or not?
return;
auto* candidates = m_nameContext.getTypes(typeReference.name());
if (candidates == nullptr) {
setError();
return;
}
for (auto& candidate : *candidates)
Visitor::visit(candidate);
if (auto result = resolveTypeOverloadImpl(*candidates, typeReference.typeArguments()))
typeReference.setResolvedType(*result);
else {
setError();
return;
}
}
void NameResolver::visit(AST::FunctionDefinition& functionDefinition)
{
NameContext newNameContext(&m_nameContext);
NameResolver newNameResolver(newNameContext);
newNameResolver.setCurrentFunctionDefinition(m_currentFunction);
checkErrorAndVisit(functionDefinition.type());
for (auto& parameter : functionDefinition.parameters())
newNameResolver.checkErrorAndVisit(parameter);
newNameResolver.checkErrorAndVisit(functionDefinition.block());
}
void NameResolver::visit(AST::Block& block)
{
NameContext nameContext(&m_nameContext);
NameResolver newNameResolver(nameContext);
newNameResolver.setCurrentFunctionDefinition(m_currentFunction);
newNameResolver.Visitor::visit(block);
}
void NameResolver::visit(AST::IfStatement& ifStatement)
{
checkErrorAndVisit(ifStatement.conditional());
NameContext nameContext(&m_nameContext);
NameResolver newNameResolver(nameContext);
newNameResolver.setCurrentFunctionDefinition(m_currentFunction);
newNameResolver.checkErrorAndVisit(ifStatement.body());
if (ifStatement.elseBody()) {
NameContext nameContext(&m_nameContext);
NameResolver newNameResolver(nameContext);
newNameResolver.setCurrentFunctionDefinition(m_currentFunction);
newNameResolver.checkErrorAndVisit(*ifStatement.elseBody());
}
}
void NameResolver::visit(AST::WhileLoop& whileLoop)
{
checkErrorAndVisit(whileLoop.conditional());
NameContext nameContext(&m_nameContext);
NameResolver newNameResolver(nameContext);
newNameResolver.setCurrentFunctionDefinition(m_currentFunction);
newNameResolver.checkErrorAndVisit(whileLoop.body());
}
void NameResolver::visit(AST::DoWhileLoop& whileLoop)
{
NameContext nameContext(&m_nameContext);
NameResolver newNameResolver(nameContext);
newNameResolver.setCurrentFunctionDefinition(m_currentFunction);
newNameResolver.checkErrorAndVisit(whileLoop.body());
checkErrorAndVisit(whileLoop.conditional());
}
void NameResolver::visit(AST::ForLoop& forLoop)
{
NameContext nameContext(&m_nameContext);
NameResolver newNameResolver(nameContext);
newNameResolver.setCurrentFunctionDefinition(m_currentFunction);
newNameResolver.Visitor::visit(forLoop);
}
void NameResolver::visit(AST::VariableDeclaration& variableDeclaration)
{
if (!m_nameContext.add(variableDeclaration)) {
setError();
return;
}
Visitor::visit(variableDeclaration);
}
void NameResolver::visit(AST::VariableReference& variableReference)
{
if (variableReference.variable())
return;
if (auto* variable = m_nameContext.getVariable(variableReference.name()))
variableReference.setVariable(*variable);
else {
setError();
return;
}
}
void NameResolver::visit(AST::Return& returnStatement)
{
ASSERT(m_currentFunction);
returnStatement.setFunction(m_currentFunction);
Visitor::visit(returnStatement);
}
void NameResolver::visit(AST::PropertyAccessExpression& propertyAccessExpression)
{
if (auto* getterFunctions = m_nameContext.getFunctions(propertyAccessExpression.getterFunctionName()))
propertyAccessExpression.setPossibleGetterOverloads(*getterFunctions);
if (auto* setterFunctions = m_nameContext.getFunctions(propertyAccessExpression.setterFunctionName()))
propertyAccessExpression.setPossibleSetterOverloads(*setterFunctions);
if (auto* anderFunctions = m_nameContext.getFunctions(propertyAccessExpression.anderFunctionName()))
propertyAccessExpression.setPossibleAnderOverloads(*anderFunctions);
Visitor::visit(propertyAccessExpression);
}
void NameResolver::visit(AST::DotExpression& dotExpression)
{
if (is<AST::VariableReference>(dotExpression.base())) {
auto baseName = downcast<AST::VariableReference>(dotExpression.base()).name();
if (auto enumerationTypes = m_nameContext.getTypes(baseName)) {
ASSERT(enumerationTypes->size() == 1);
AST::NamedType& type = (*enumerationTypes)[0];
if (is<AST::EnumerationDefinition>(type)) {
AST::EnumerationDefinition& enumerationDefinition = downcast<AST::EnumerationDefinition>(type);
auto memberName = dotExpression.fieldName();
if (auto* member = enumerationDefinition.memberByName(memberName)) {
Lexer::Token origin = dotExpression.origin();
auto enumerationMemberLiteral = AST::EnumerationMemberLiteral::wrap(WTFMove(origin), WTFMove(baseName), WTFMove(memberName), enumerationDefinition, *member);
AST::replaceWith<AST::EnumerationMemberLiteral>(dotExpression, WTFMove(enumerationMemberLiteral));
return;
}
setError();
return;
}
}
}
Visitor::visit(dotExpression);
}
void NameResolver::visit(AST::CallExpression& callExpression)
{
if (!callExpression.hasOverloads()) {
if (auto* functions = m_nameContext.getFunctions(callExpression.name()))
callExpression.setOverloads(*functions);
else {
if (auto* types = m_nameContext.getTypes(callExpression.name())) {
if (types->size() == 1) {
if (auto* functions = m_nameContext.getFunctions("operator cast"_str)) {
callExpression.setCastData((*types)[0].get());
callExpression.setOverloads(*functions);
}
}
}
}
}
if (!callExpression.hasOverloads()) {
setError();
return;
}
Visitor::visit(callExpression);
}
void NameResolver::visit(AST::EnumerationMemberLiteral& enumerationMemberLiteral)
{
if (enumerationMemberLiteral.enumerationMember())
return;
if (auto enumerationTypes = m_nameContext.getTypes(enumerationMemberLiteral.left())) {
ASSERT(enumerationTypes->size() == 1);
AST::NamedType& type = (*enumerationTypes)[0];
if (is<AST::EnumerationDefinition>(type)) {
AST::EnumerationDefinition& enumerationDefinition = downcast<AST::EnumerationDefinition>(type);
if (auto* member = enumerationDefinition.memberByName(enumerationMemberLiteral.right())) {
enumerationMemberLiteral.setEnumerationMember(enumerationDefinition, *member);
return;
}
}
}
setError();
}
// FIXME: https://bugs.webkit.org/show_bug.cgi?id=198167 Make sure all the names have been resolved.
bool resolveNamesInTypes(Program& program, NameResolver& nameResolver)
{
for (auto& typeDefinition : program.typeDefinitions()) {
nameResolver.checkErrorAndVisit(typeDefinition);
if (nameResolver.error())
return false;
}
for (auto& structureDefinition : program.structureDefinitions()) {
nameResolver.checkErrorAndVisit(structureDefinition);
if (nameResolver.error())
return false;
}
for (auto& enumerationDefinition : program.enumerationDefinitions()) {
nameResolver.checkErrorAndVisit(enumerationDefinition);
if (nameResolver.error())
return false;
}
for (auto& nativeTypeDeclaration : program.nativeTypeDeclarations()) {
nameResolver.checkErrorAndVisit(nativeTypeDeclaration);
if (nameResolver.error())
return false;
}
return true;
}
bool resolveNamesInFunctions(Program& program, NameResolver& nameResolver)
{
for (auto& functionDefinition : program.functionDefinitions()) {
nameResolver.setCurrentFunctionDefinition(&functionDefinition);
nameResolver.checkErrorAndVisit(functionDefinition);
if (nameResolver.error())
return false;
}
nameResolver.setCurrentFunctionDefinition(nullptr);
for (auto& nativeFunctionDeclaration : program.nativeFunctionDeclarations()) {
nameResolver.checkErrorAndVisit(nativeFunctionDeclaration);
if (nameResolver.error())
return false;
}
return true;
}
} // namespace WHLSL
} // namespace WebCore
#endif // ENABLE(WEBGPU)