- Timestamp:
- Oct 16, 2023, 3:04:18 PM (14 months ago)
- Branches:
- master
- Children:
- 278e162
- Parents:
- 54e59dd (diff), e14d169 (diff)
Note: this is a merge changeset, the changes displayed below correspond to the merge itself.
Use the(diff)
links above to see all the changes relative to each parent. - File:
-
- 1 edited
Legend:
- Unmodified
- Added
- Removed
-
src/GenPoly/BoxNew.cpp
r54e59dd r4d2d7e27 376 376 ast::Expr const * postvisit( ast::AddressExpr const * expr ); 377 377 ast::ReturnStmt const * previsit( ast::ReturnStmt const * stmt ); 378 void previsit( ast::PointerType const * type );379 void previsit( ast::FunctionType const * type );380 378 381 379 void beginScope(); … … 386 384 // return value. 387 385 388 /// Pass the extra type parameters from polymorphic generic arguments or 389 /// return types into a function application. 390 ast::vector<ast::Expr>::iterator passArgTypeVars( 391 ast::ApplicationExpr * expr, ast::Type const * parmType, 392 ast::Type const * argBaseType, ast::vector<ast::Expr>::iterator arg, 393 const TypeVarMap & exprTyVars, std::set<std::string> & seenTypes ); 394 /// Passes extra type parameters into a polymorphic function application. 386 /// Passes extra layout arguments for sized polymorphic type parameters. 395 387 ast::vector<ast::Expr>::iterator passTypeVars( 396 388 ast::ApplicationExpr * expr, 397 ast::Type const * polyRetType, 398 ast::FunctionType const * funcType, 399 const TypeVarMap & exprTyVars ); 389 ast::FunctionType const * funcType ); 400 390 /// Wraps a function application with a new temporary for the 401 391 /// out-parameter return value. … … 561 551 562 552 ast::FunctionDecl const * CallAdapter::previsit( ast::FunctionDecl const * decl ) { 553 // Prevent type declaration information from leaking out. 554 GuardScope( scopeTypeVars ); 555 563 556 if ( nullptr == decl->stmts ) { 564 // This may keep TypeDecls we don't ever want from sneaking in.565 // Not visiting child nodes might just be faster.566 GuardScope( scopeTypeVars );567 557 return decl; 568 558 } 569 559 570 GuardScope( scopeTypeVars );571 560 GuardValue( retval ); 572 561 … … 695 684 696 685 assert( typeSubs ); 697 ast::Type const * concRetType = replaceWithConcrete( dynRetType, *typeSubs );698 // Used to use dynRetType instead of concRetType; this changed so that699 // the correct type parameters are passed for return types (it should be700 // the concrete type's parameters, not the formal type's).701 686 ast::vector<ast::Expr>::iterator argIt = 702 passTypeVars( mutExpr, concRetType, function, exprTypeVars);687 passTypeVars( mutExpr, function ); 703 688 addInferredParams( mutExpr, argIt, function, exprTypeVars ); 704 689 … … 778 763 } 779 764 780 void CallAdapter::previsit( ast::PointerType const * type ) {781 GuardScope( scopeTypeVars );782 makeTypeVarMap( type, scopeTypeVars );783 }784 785 void CallAdapter::previsit( ast::FunctionType const * type ) {786 GuardScope( scopeTypeVars );787 makeTypeVarMap( type, scopeTypeVars );788 }789 790 765 void CallAdapter::beginScope() { 791 766 adapters.beginScope(); … … 812 787 } 813 788 814 // arg is an in/out parameter that matches the return value.815 ast::vector<ast::Expr>::iterator CallAdapter::passArgTypeVars(816 ast::ApplicationExpr * expr, ast::Type const * paramType,817 ast::Type const * argBaseType, ast::vector<ast::Expr>::iterator arg,818 const TypeVarMap & exprTypeVars, std::set<std::string> & seenTypes ) {819 ast::Type const * polyType = isPolyType( paramType, exprTypeVars );820 if ( !polyType || dynamic_cast<ast::TypeInstType const *>( polyType ) ) {821 return arg;822 }823 824 std::string typeName = Mangle::mangleType( polyType );825 if ( seenTypes.count( typeName ) ) return arg;826 827 arg = expr->args.insert( arg,828 new ast::SizeofExpr( expr->location, ast::deepCopy( argBaseType ) )829 );830 arg++;831 arg = expr->args.insert( arg,832 new ast::AlignofExpr( expr->location, ast::deepCopy( argBaseType ) )833 );834 arg++;835 if ( dynamic_cast<ast::StructInstType const *>( polyType ) ) {836 auto argBaseStructType =837 dynamic_cast<ast::StructInstType const *>( argBaseType );838 if ( nullptr == argBaseStructType ) {839 SemanticError( expr,840 "Cannot pass non-structure type for generic struct: " );841 }842 843 // Zero-length arrays are forbidden by C, so don't pass844 // offset for empty structure.845 if ( !argBaseStructType->base->members.empty() ) {846 arg = expr->args.insert( arg,847 new ast::OffsetPackExpr(848 expr->location,849 ast::deepCopy( argBaseStructType ) )850 );851 arg++;852 }853 }854 855 seenTypes.insert( typeName );856 return arg;857 }858 859 789 ast::vector<ast::Expr>::iterator CallAdapter::passTypeVars( 860 790 ast::ApplicationExpr * expr, 861 ast::Type const * polyRetType, 862 ast::FunctionType const * function, 863 const TypeVarMap & exprTypeVars ) { 791 ast::FunctionType const * function ) { 864 792 assert( typeSubs ); 865 793 ast::vector<ast::Expr>::iterator arg = expr->args.begin(); … … 880 808 new ast::AlignofExpr( expr->location, ast::deepCopy( concrete ) ) ); 881 809 arg++; 882 }883 884 // Add size/align for generic types to parameter list.885 if ( !expr->func->result ) return arg;886 ast::FunctionType const * funcType = getFunctionType( expr->func->result );887 assert( funcType );888 889 // This iterator points at first original argument.890 ast::vector<ast::Expr>::const_iterator funcArg;891 // Names for generic types we've seen.892 std::set<std::string> seenTypes;893 894 // A polymorphic return type may need to be added to the argument list.895 if ( polyRetType ) {896 assert( typeSubs );897 auto concRetType = replaceWithConcrete( polyRetType, *typeSubs );898 // TODO: This write-back may not be correct.899 arg = passArgTypeVars( expr, polyRetType, concRetType,900 arg, exprTypeVars, seenTypes );901 // Skip the return parameter in the argument list.902 funcArg = arg + 1;903 } else {904 funcArg = arg;905 }906 907 // TODO:908 // I believe this is (starts as) the number of original arguments to the909 // function with the args before funcArg all being inserted.910 ptrdiff_t argsToPass = std::distance( funcArg, expr->args.cend() );911 912 // Add type information args for presently unseen types in parameter list.913 ast::vector<ast::Type>::const_iterator funcParam = funcType->params.begin();914 // assert( funcType->params.size() == argsToPass );915 for ( ; funcParam != funcType->params.end() && 0 < argsToPass916 ; ++funcParam, --argsToPass ) {917 assert( 0 < argsToPass );918 assert( argsToPass <= (ptrdiff_t)expr->args.size() );919 ptrdiff_t index = expr->args.size() - argsToPass;920 ast::Type const * argType = expr->args[index]->result;921 if ( nullptr == argType ) continue;922 arg = passArgTypeVars( expr, *funcParam, argType,923 arg, exprTypeVars, seenTypes );924 810 } 925 811 return arg; … … 1523 1409 } 1524 1410 1525 ast::ObjectDecl * makePtr(1526 CodeLocation const & location, std::string const & name ) {1527 return new ast::ObjectDecl( location, name,1528 new ast::PointerType( makeSizeAlignType() ),1529 nullptr, ast::Storage::Classes(), ast::Linkage::C, nullptr );1530 }1531 1532 1411 ast::FunctionDecl const * DeclAdapter::previsit( ast::FunctionDecl const * decl ) { 1533 1412 TypeVarMap localTypeVars = { ast::TypeData() }; … … 1584 1463 mutDecl->assertions.clear(); 1585 1464 1586 // Add size/align for generic parameter types to parameter list.1587 std::set<std::string> seenTypes;1588 ast::vector<ast::DeclWithType> otypeParams;1589 for ( ast::ptr<ast::DeclWithType> & funcParam : mutDecl->params ) {1590 ast::Type const * polyType = isPolyType( funcParam->get_type(), localTypeVars );1591 if ( !polyType || dynamic_cast<ast::TypeInstType const *>( polyType ) ) {1592 continue;1593 }1594 std::string typeName = Mangle::mangleType( polyType );1595 if ( seenTypes.count( typeName ) ) continue;1596 seenTypes.insert( typeName );1597 1598 auto sizeParam = makeObj( funcParam->location, sizeofName( typeName ) );1599 otypeParams.emplace_back( sizeParam );1600 1601 auto alignParam = makeObj( funcParam->location, alignofName( typeName ) );1602 otypeParams.emplace_back( alignParam );1603 1604 // Zero-length arrays are illegal in C, so empty structs have no1605 // offset array.1606 if ( auto * polyStruct =1607 dynamic_cast<ast::StructInstType const *>( polyType ) ;1608 polyStruct && !polyStruct->base->members.empty() ) {1609 auto offsetParam = makePtr( funcParam->location, offsetofName( typeName ) );1610 otypeParams.emplace_back( offsetParam );1611 }1612 }1613 1614 1465 // Prepend each argument group. From last group to first. addAdapters 1615 1466 // does do the same, it just does it itself and see all other parameters. 1616 1467 spliceBegin( mutDecl->params, inferredParams ); 1617 spliceBegin( mutDecl->params, otypeParams );1618 1468 spliceBegin( mutDecl->params, layoutParams ); 1619 1469 addAdapters( mutDecl, localTypeVars ); … … 1728 1578 PolyGenericCalculator(); 1729 1579 1730 void previsit( ast::ObjectDecl const * decl );1731 1580 void previsit( ast::FunctionDecl const * decl ); 1732 1581 void previsit( ast::TypedefDecl const * decl ); … … 1735 1584 ast::StructDecl const * previsit( ast::StructDecl const * decl ); 1736 1585 ast::UnionDecl const * previsit( ast::UnionDecl const * decl ); 1737 void previsit( ast::PointerType const * type );1738 void previsit( ast::FunctionType const * type );1739 1586 ast::DeclStmt const * previsit( ast::DeclStmt const * stmt ); 1740 1587 ast::Expr const * postvisit( ast::MemberExpr const * expr ); … … 1772 1619 /// adding the type variables from the provided type. 1773 1620 void beginTypeScope( ast::Type const * ); 1774 /// Enters a new scope for known layouts and offsets, and queues exit calls.1775 void beginGenericScope();1776 1621 1777 1622 /// Set of generic type layouts known in the current scope, … … 1785 1630 /// If the argument of an AddressExpr is MemberExpr, it is stored here. 1786 1631 ast::MemberExpr const * addrMember = nullptr; 1787 /// Used to avoid recursing too deep in type declarations.1788 bool expect_func_type = false;1789 1632 }; 1790 1633 … … 1808 1651 } 1809 1652 1810 void PolyGenericCalculator::previsit( ast::ObjectDecl const * decl ) { 1653 void PolyGenericCalculator::previsit( ast::FunctionDecl const * decl ) { 1654 GuardScope( *this ); 1811 1655 beginTypeScope( decl->type ); 1812 }1813 1814 void PolyGenericCalculator::previsit( ast::FunctionDecl const * decl ) {1815 beginGenericScope();1816 beginTypeScope( decl->type );1817 1818 // TODO: Going though dec->params does not work for some reason.1819 for ( ast::ptr<ast::Type> const & funcParam : decl->type->params ) {1820 // Condition here duplicates that in `DeclAdapter::previsit( FunctionDecl const * )`1821 ast::Type const * polyType = isPolyType( funcParam, scopeTypeVars );1822 if ( polyType && !dynamic_cast<ast::TypeInstType const *>( polyType ) ) {1823 knownLayouts.insert( Mangle::mangleType( polyType ) );1824 }1825 }1826 1656 } 1827 1657 … … 1884 1714 } 1885 1715 1886 void PolyGenericCalculator::previsit( ast::PointerType const * type ) {1887 beginTypeScope( type );1888 }1889 1890 void PolyGenericCalculator::previsit( ast::FunctionType const * type ) {1891 beginTypeScope( type );1892 1893 GuardValue( expect_func_type );1894 GuardScope( *this );1895 1896 // The other functions type we will see in this scope are probably1897 // function parameters they don't help us with the layout and offsets so1898 // don't mark them as known in this scope.1899 expect_func_type = false;1900 1901 // Make sure that any type information passed into the function is1902 // accounted for.1903 for ( ast::ptr<ast::Type> const & funcParam : type->params ) {1904 // Condition here duplicates that in `DeclAdapter::previsit( FunctionDecl const * )`1905 ast::Type const * polyType = isPolyType( funcParam, scopeTypeVars );1906 if ( polyType && !dynamic_cast<ast::TypeInstType const *>( polyType ) ) {1907 knownLayouts.insert( Mangle::mangleType( polyType ) );1908 }1909 }1910 }1911 1912 //void PolyGenericCalculator::previsit( ast::DeclStmt const * stmt ) {1913 1716 ast::DeclStmt const * PolyGenericCalculator::previsit( ast::DeclStmt const * stmt ) { 1914 1717 ast::ObjectDecl const * decl = stmt->decl.as<ast::ObjectDecl>(); … … 2400 2203 } 2401 2204 2402 void PolyGenericCalculator::beginGenericScope() {2403 GuardScope( *this );2404 // We expect the first function type see to be the type relating to this2405 // scope but any further type is probably some unrelated function pointer2406 // keep track of whrich is the first.2407 GuardValue( expect_func_type ) = true;2408 }2409 2410 2205 // -------------------------------------------------------------------------- 2411 /// No common theme found.2206 /// Removes unneeded or incorrect type information. 2412 2207 /// * Replaces initialization of polymorphic values with alloca. 2413 2208 /// * Replaces declaration of dtype/ftype with appropriate void expression.
Note: See TracChangeset
for help on using the changeset viewer.