| // Copyright (c) 2019, the Dart project authors. Please see the AUTHORS file |
| // for details. All rights reserved. Use of this source code is governed by a |
| // BSD-style license that can be found in the LICENSE file. |
| |
| // Generates the API tables used by DartFuzz. Automatically generating these |
| // tables is less error-prone than generating such tables by hand. Furthermore, |
| // it simplifies regenerating the table when the libraries change. |
| // |
| // Usage: |
| // dart gen_api_table.dart > dartfuzz_api_table.dart |
| // |
| // Then send out modified dartfuzz_api_table.dart for review together |
| // with a modified dartfuzz.dart that increases the version. |
| |
| import 'package:analyzer/dart/analysis/results.dart'; |
| import 'package:analyzer/dart/analysis/session.dart'; |
| import 'package:analyzer/dart/element/element.dart'; |
| import 'package:analyzer/dart/element/nullability_suffix.dart'; |
| import 'package:analyzer/dart/element/type.dart'; |
| |
| import 'gen_util.dart'; |
| |
| // Enum for different restrictions on parameters for library methods. |
| // none - No restriction on the corresponding parameter. |
| // small - Corresponding parameter should be a small value. |
| // This enum has an equivalent enum in the generated Dartfuzz API table. |
| enum Restriction { none, small } |
| |
| // Class that represents Dart library methods. |
| // |
| // Proto is a list (of Strings that represent DartTypes) whose first element is |
| // the DartType of the receiver (DartType.VOID if none). The remaining elements |
| // are DartTypes of the parameters. The second element is DartType.VOID if there |
| // are no parameters. |
| // This class has an equivalent class in the generated Dartfuzz API table. |
| class DartLib { |
| final String name; |
| final List<String> proto; |
| final List<Restriction>? restrictions; |
| final bool isMethod; |
| const DartLib(this.name, this.proto, this.restrictions, this.isMethod); |
| } |
| |
| // Constants for strings corresponding to the DartType. |
| const abstractClassInstantiationErrorEncoding = |
| 'DartType.ABSTRACTCLASSINSTANTIATIONERROR'; |
| const argumentErrorEncoding = 'DartType.ARGUMENTERROR'; |
| const assertionErrorEncoding = 'DartType.ERROR'; |
| const boolEncoding = 'DartType.BOOL'; |
| const byteDataEncoding = 'DartType.BYTEDATA'; |
| const castErrorEncoding = 'DartType.CASTERROR'; |
| const concurrentModificationErrorEncoding = |
| 'DartType.CONCURRENTMODIFICATIONERROR'; |
| const cyclicInitializationErrorEncoding = 'DartType.CYCLICINITIALIZATIONERROR'; |
| const deprecatedEncoding = 'DartType.DEPRECATED'; |
| const doubleEncoding = 'DartType.DOUBLE'; |
| const endianEncoding = 'DartType.ENDIAN'; |
| const errorEncoding = 'DartType.ERROR'; |
| const exceptionEncoding = 'DartType.EXCEPTION'; |
| const expandoDoubleEncoding = 'DartType.EXPANDO_DOUBLE'; |
| const expandoIntEncoding = 'DartType.EXPANDO_INT'; |
| const fallThroughErrorEncoding = 'DartType.FALLTHROUGHERROR'; |
| const float32ListEncoding = 'DartType.FLOAT32LIST'; |
| const float32x4Encoding = 'DartType.FLOAT32X4'; |
| const float32x4ListEncoding = 'DartType.FLOAT32X4LIST'; |
| const float64ListEncoding = 'DartType.FLOAT64LIST'; |
| const float64x2Encoding = 'DartType.FLOAT64X2'; |
| const float64x2ListEncoding = 'DartType.FLOAT64X2LIST'; |
| const formatExceptionEncoding = 'DartType.FORMATEXCEPTION'; |
| const indexErrorEncoding = 'DartType.INDEXERROR'; |
| const int16ListEncoding = 'DartType.INT16LIST'; |
| const int32ListEncoding = 'DartType.INT32LIST'; |
| const int32x4Encoding = 'DartType.INT32X4'; |
| const int32x4ListEncoding = 'DartType.INT32X4LIST'; |
| const int64ListEncoding = 'DartType.INT64LIST'; |
| const int8ListEncoding = 'DartType.INT8LIST'; |
| const intEncoding = 'DartType.INT'; |
| const integerDivisionByZeroExceptionEncoding = |
| 'DartType.INTEGERDIVISIONBYZEROEXCEPTION'; |
| const listIntEncoding = 'DartType.LIST_INT'; |
| const mapEntryIntStringEncoding = 'DartType.MAPENTRY_INT_STRING'; |
| const mapIntStringEncoding = 'DartType.MAP_INT_STRING'; |
| const nullEncoding = 'DartType.NULL'; |
| const nullThrownErrorEncoding = 'DartType.NULLTHROWNERROR'; |
| const numEncoding = 'DartType.NUM'; |
| const provisionalEncoding = 'DartType.PROVISIONAL'; |
| const rangeErrorEncoding = 'DartType.RANGEERROR'; |
| const regExpEncoding = 'DartType.REGEXP'; |
| const runeIteratorEncoding = 'DartType.RUNEITERATOR'; |
| const runesEncoding = 'DartType.RUNES'; |
| const setIntEncoding = 'DartType.SET_INT'; |
| const stackOverflowErrorEncoding = 'DartType.STACKOVERFLOWERROR'; |
| const stateErrorEncoding = 'DartType.STATEERROR'; |
| const stringBufferEncoding = 'DartType.STRINGBUFFER'; |
| const stringEncoding = 'DartType.STRING'; |
| const symbolEncoding = 'DartType.SYMBOL'; |
| const typeErrorEncoding = 'DartType.TYPEERROR'; |
| const uint16ListEncoding = 'DartType.UINT16LIST'; |
| const uint32ListEncoding = 'DartType.UINT32LIST'; |
| const uint64ListEncoding = 'DartType.UINT64LIST'; |
| const uint8ClampedListEncoding = 'DartType.UINT8CLAMPEDLIST'; |
| const uint8ListEncoding = 'DartType.UINT8LIST'; |
| const unimplementedErrorEncoding = 'DartType.UNIMPLEMENTEDERROR'; |
| const unsupportedErrorEncoding = 'DartType.UNSUPPORTEDERROR'; |
| const voidEncoding = 'DartType.VOID'; |
| |
| // Constants for the library methods lists' names in dart_api_table.dart. |
| final abstractClassInstantiationErrorLibs = |
| 'abstractClassInstantiationErrorLibs'; |
| final argumentErrorLibs = 'argumentErrorLibs'; |
| final assertionErrorLibs = 'assertionErrorLibs'; |
| final boolLibs = 'boolLibs'; |
| final byteDataLibs = 'byteDataLibs'; |
| final castErrorLibs = 'castErrorLibs'; |
| final concurrentModificationErrorLibs = 'concurrentModificationErrorLibs'; |
| final cyclicInitializationErrorLibs = 'cyclicInitializationErrorLibs'; |
| final deprecatedLibs = 'deprecatedLibs'; |
| final doubleLibs = 'doubleLibs'; |
| final endianLibs = 'endianLibs'; |
| final errorLibs = 'errorLibs'; |
| final exceptionLibs = 'exceptionLibs'; |
| final expandoDoubleLibs = 'expandoDoubleLibs'; |
| final expandoIntLibs = 'expandoIntLibs'; |
| final fallThroughErrorLibs = 'fallThroughErrorLibs'; |
| final float32ListLibs = 'float32ListLibs'; |
| final float32x4Libs = 'float32x4Libs'; |
| final float32x4ListLibs = 'float32x4ListLibs'; |
| final float64ListLibs = 'float64ListLibs'; |
| final float64x2Libs = 'float64x2Libs'; |
| final float64x2ListLibs = 'float64x2ListLibs'; |
| final formatExceptionLibs = 'formatExceptionLibs'; |
| final indexErrorLibs = 'indexErrorLibs'; |
| final int16ListLibs = 'int16ListLibs'; |
| final int32ListLibs = 'int32ListLibs'; |
| final int32x4Libs = 'int32x4Libs'; |
| final int32x4ListLibs = 'int32x4ListLibs'; |
| final int64ListLibs = 'int64ListLibs'; |
| final int8ListLibs = 'int8ListLibs'; |
| final intLibs = 'intLibs'; |
| final integerDivisionByZeroExceptionLibs = 'integerDivisionByZeroExceptionLibs'; |
| final listLibs = 'listLibs'; |
| final mapEntryIntStringLibs = 'mapEntryIntStringLibs'; |
| final mapLibs = 'mapLibs'; |
| final nullLibs = 'nullLibs'; |
| final nullThrownErrorLibs = 'nullThrownErrorLibs'; |
| final numLibs = 'numLibs'; |
| final provisionalLibs = 'provisionalLibs'; |
| final rangeErrorLibs = 'rangeErrorLibs'; |
| final regExpLibs = 'regExpLibs'; |
| final runeIteratorLibs = 'runeIteratorLibs'; |
| final runesLibs = 'runesLibs'; |
| final setLibs = 'setLibs'; |
| final stackOverflowErrorLibs = 'stackOverflowErrorLibs'; |
| final stateErrorLibs = 'stateErrorLibs'; |
| final stringBufferLibs = 'stringBufferLibs'; |
| final stringLibs = 'stringLibs'; |
| final symbolLibs = 'symbolLibs'; |
| final typeErrorLibs = 'typeErrorLibs'; |
| final uint16ListLibs = 'uint16ListLibs'; |
| final uint32ListLibs = 'uint32ListLibs'; |
| final uint64ListLibs = 'uint64ListLibs'; |
| final uint8ClampedListLibs = 'uint8ClampedListLibs'; |
| final uint8ListLibs = 'uint8ListLibs'; |
| final unimplementedErrorLibs = 'unimplementedErrorLibs'; |
| final unsupportedErrorLibs = 'unsupportedErrorLibs'; |
| final voidLibs = 'voidLibs'; |
| |
| // Map from the DartType (string) to the name of the library methods list. |
| final Map<String, String> typeToLibraryMethodsListName = () { |
| var encodings = <String, String>{ |
| abstractClassInstantiationErrorEncoding: |
| abstractClassInstantiationErrorLibs, |
| argumentErrorEncoding: argumentErrorLibs, |
| assertionErrorEncoding: assertionErrorLibs, |
| boolEncoding: boolLibs, |
| byteDataEncoding: byteDataLibs, |
| castErrorEncoding: castErrorLibs, |
| concurrentModificationErrorEncoding: concurrentModificationErrorLibs, |
| cyclicInitializationErrorEncoding: cyclicInitializationErrorLibs, |
| deprecatedEncoding: deprecatedLibs, |
| doubleEncoding: doubleLibs, |
| endianEncoding: endianLibs, |
| errorEncoding: errorLibs, |
| exceptionEncoding: exceptionLibs, |
| expandoDoubleEncoding: expandoDoubleLibs, |
| expandoIntEncoding: expandoIntLibs, |
| fallThroughErrorEncoding: fallThroughErrorLibs, |
| float32ListEncoding: float32ListLibs, |
| float32x4Encoding: float32x4Libs, |
| float32x4ListEncoding: float32x4ListLibs, |
| float64ListEncoding: float64ListLibs, |
| float64x2Encoding: float64x2Libs, |
| float64x2ListEncoding: float64x2ListLibs, |
| formatExceptionEncoding: formatExceptionLibs, |
| indexErrorEncoding: indexErrorLibs, |
| int16ListEncoding: int16ListLibs, |
| int32ListEncoding: int32ListLibs, |
| int32x4Encoding: int32x4Libs, |
| int32x4ListEncoding: int32x4ListLibs, |
| int64ListEncoding: int64ListLibs, |
| int8ListEncoding: int8ListLibs, |
| intEncoding: intLibs, |
| integerDivisionByZeroExceptionEncoding: integerDivisionByZeroExceptionLibs, |
| listIntEncoding: listLibs, |
| mapEntryIntStringEncoding: mapEntryIntStringLibs, |
| mapIntStringEncoding: mapLibs, |
| nullEncoding: nullLibs, |
| nullThrownErrorEncoding: nullThrownErrorLibs, |
| numEncoding: numLibs, |
| provisionalEncoding: provisionalLibs, |
| rangeErrorEncoding: rangeErrorLibs, |
| regExpEncoding: regExpLibs, |
| runeIteratorEncoding: runeIteratorLibs, |
| runesEncoding: runesLibs, |
| setIntEncoding: setLibs, |
| stackOverflowErrorEncoding: stackOverflowErrorLibs, |
| stateErrorEncoding: stateErrorLibs, |
| stringBufferEncoding: stringBufferLibs, |
| stringEncoding: stringLibs, |
| symbolEncoding: symbolLibs, |
| typeErrorEncoding: typeErrorLibs, |
| uint16ListEncoding: uint16ListLibs, |
| uint32ListEncoding: uint32ListLibs, |
| uint64ListEncoding: uint64ListLibs, |
| uint8ClampedListEncoding: uint8ClampedListLibs, |
| uint8ListEncoding: uint8ListLibs, |
| unimplementedErrorEncoding: unimplementedErrorLibs, |
| unsupportedErrorEncoding: unsupportedErrorLibs, |
| voidEncoding: voidLibs, |
| }; |
| var result = <String, String>{}; |
| encodings.forEach((key, value) { |
| result[key] = value; |
| result[key + "_NULLABLE"] = value + "Nullable"; |
| }); |
| return result; |
| }(); |
| |
| // Map from return type encoding to list of recognized methods with that |
| // return type. |
| final Map<String, List<DartLib>> typeToLibraryMethodsList = () { |
| var encodings = <String>[ |
| abstractClassInstantiationErrorEncoding, |
| argumentErrorEncoding, |
| assertionErrorEncoding, |
| boolEncoding, |
| byteDataEncoding, |
| castErrorEncoding, |
| concurrentModificationErrorEncoding, |
| cyclicInitializationErrorEncoding, |
| deprecatedEncoding, |
| doubleEncoding, |
| endianEncoding, |
| errorEncoding, |
| exceptionEncoding, |
| expandoDoubleEncoding, |
| expandoIntEncoding, |
| fallThroughErrorEncoding, |
| float32ListEncoding, |
| float32x4Encoding, |
| float32x4ListEncoding, |
| float64ListEncoding, |
| float64x2Encoding, |
| float64x2ListEncoding, |
| formatExceptionEncoding, |
| indexErrorEncoding, |
| int16ListEncoding, |
| int32ListEncoding, |
| int32x4Encoding, |
| int32x4ListEncoding, |
| int64ListEncoding, |
| int8ListEncoding, |
| intEncoding, |
| integerDivisionByZeroExceptionEncoding, |
| listIntEncoding, |
| mapEntryIntStringEncoding, |
| mapIntStringEncoding, |
| nullEncoding, |
| nullThrownErrorEncoding, |
| numEncoding, |
| provisionalEncoding, |
| rangeErrorEncoding, |
| regExpEncoding, |
| runeIteratorEncoding, |
| runesEncoding, |
| setIntEncoding, |
| stackOverflowErrorEncoding, |
| stateErrorEncoding, |
| stringBufferEncoding, |
| stringEncoding, |
| symbolEncoding, |
| typeErrorEncoding, |
| uint16ListEncoding, |
| uint32ListEncoding, |
| uint64ListEncoding, |
| uint8ClampedListEncoding, |
| uint8ListEncoding, |
| unimplementedErrorEncoding, |
| unsupportedErrorEncoding, |
| voidEncoding, |
| ]; |
| var result = <String, List<DartLib>>{}; |
| for (var encoding in encodings) { |
| result[encoding] = <DartLib>[]; |
| result[encoding + "_NULLABLE"] = <DartLib>[]; |
| } |
| return result; |
| }(); |
| |
| final typedDataFloatTypes = [ |
| float32ListEncoding, |
| float32x4Encoding, |
| float32x4ListEncoding, |
| float64ListEncoding, |
| float64x2Encoding, |
| float64x2ListEncoding |
| ]; |
| |
| void main() async { |
| final session = GenUtil.createAnalysisSession(); |
| |
| // Visit libraries for table generation. |
| await visitLibraryAtUri(session, 'dart:async'); |
| await visitLibraryAtUri(session, 'dart:cli'); |
| await visitLibraryAtUri(session, 'dart:collection'); |
| await visitLibraryAtUri(session, 'dart:convert'); |
| await visitLibraryAtUri(session, 'dart:core'); |
| await visitLibraryAtUri(session, 'dart:io'); |
| await visitLibraryAtUri(session, 'dart:isolate'); |
| await visitLibraryAtUri(session, 'dart:math'); |
| await visitLibraryAtUri(session, 'dart:typed_data'); |
| |
| // Generate the tables in a stand-alone Dart class. |
| dumpHeader(); |
| dumpTypeToLibraryMethodMap(); |
| dumpTypedDataFloatTypes(); |
| for (var key in typeToLibraryMethodsList.keys.toList()..sort()) { |
| if (typeToLibraryMethodsList[key]!.isNotEmpty) { |
| // Only output library methods lists that are non-empty. |
| dumpTable( |
| typeToLibraryMethodsListName[key]!, typeToLibraryMethodsList[key]!); |
| } |
| } |
| dumpFooter(); |
| } |
| |
| Future<void> visitLibraryAtUri(AnalysisSession session, String uri) async { |
| final libPath = session.uriConverter.uriToPath(Uri.parse(uri)); |
| var result = await session.getResolvedLibrary(libPath!); |
| if (result is ResolvedLibraryResult) { |
| visitLibrary(result.element); |
| } else { |
| throw StateError('Unable to resolve "$uri"'); |
| } |
| } |
| |
| void visitLibrary(LibraryElement library) { |
| // This uses the element model to traverse the code. The element model |
| // represents the semantic structure of the code. A library consists of |
| // one or more compilation units. |
| for (var unit in library.units) { |
| visitCompilationUnit(unit); |
| } |
| } |
| |
| void visitCompilationUnit(CompilationUnitElement unit) { |
| // Each compilation unit contains elements for all of the top-level |
| // declarations in a single file, such as variables, functions, and |
| // classes. Note that `types` only returns classes. You can use |
| // `mixins` to visit mixins, `enums` to visit enum, `functionTypeAliases` |
| // to visit typedefs, etc. |
| for (var variable in unit.topLevelVariables) { |
| if (variable.isPublic) { |
| addToTable(typeString(variable.type), variable.name, |
| [voidEncoding, voidEncoding], |
| isMethod: false); |
| } |
| } |
| for (var function in unit.functions) { |
| if (function.isPublic && !function.isOperator) { |
| addToTable(typeString(function.returnType), function.name, |
| protoString(null, function.parameters)); |
| } |
| } |
| for (var classElement in unit.classes) { |
| if (classElement.isPublic) { |
| visitClass(classElement); |
| } |
| } |
| } |
| |
| void visitClass(ClassElement classElement) { |
| // Classes that cause too many false divergences. |
| if (classElement.name == 'ProcessInfo' || |
| classElement.name == 'Platform' || |
| classElement.name.startsWith('FileSystem')) { |
| return; |
| } |
| // Every class element contains elements for the members, viz. `methods` visits |
| // methods, `fields` visits fields, `accessors` visits getters and setters, etc. |
| // There are also accessors to get the superclass, mixins, interfaces, type |
| // parameters, etc. |
| for (var constructor in classElement.constructors) { |
| if (constructor.isPublic && |
| constructor.isFactory && |
| constructor.name.isNotEmpty) { |
| addToTable( |
| typeString(classElement.thisType), |
| '${classString(classElement)}.${constructor.name}', |
| protoString(null, constructor.parameters)); |
| } |
| } |
| for (var method in classElement.methods) { |
| if (method.isPublic && !method.isOperator) { |
| if (method.isStatic) { |
| addToTable( |
| typeString(method.returnType), |
| '${classString(classElement)}.${method.name}', |
| protoString(null, method.parameters)); |
| } else { |
| addToTable(typeString(method.returnType), method.name, |
| protoString(classElement.thisType, method.parameters)); |
| } |
| } |
| } |
| for (var accessor in classElement.accessors) { |
| if (accessor.isPublic && accessor.isGetter) { |
| var variable = accessor.variable; |
| if (accessor.isStatic) { |
| addToTable( |
| typeString(variable.type), |
| '${classElement.name}.${variable.name}', |
| [voidEncoding, voidEncoding], |
| isMethod: false); |
| } else { |
| addToTable(typeString(variable.type), variable.name, |
| [typeString(classElement.thisType), voidEncoding], |
| isMethod: false); |
| } |
| } |
| } |
| } |
| |
| // Function that returns the explicit class name. |
| String classString(ClassElement classElement) { |
| switch (typeString(classElement.thisType)) { |
| case setIntEncoding: |
| return 'Set<int>'; |
| case listIntEncoding: |
| return 'List<int>'; |
| case mapIntStringEncoding: |
| return 'Map<int, String>'; |
| default: |
| return classElement.name; |
| } |
| } |
| |
| // Types are represented by an instance of `DartType`. For classes, the type |
| // will be an instance of `InterfaceType`, which will provide access to the |
| // defining (class) element, as well as any type arguments. |
| String typeString(DartType type) { |
| String result = typeStringHelper(type); |
| if (result == "?") { |
| return result; |
| } |
| if (type.nullabilitySuffix == NullabilitySuffix.none) { |
| return result; |
| } |
| return result + "_NULLABLE"; |
| } |
| |
| String typeStringHelper(DartType type) { |
| if (type.isDartCoreBool) { |
| return boolEncoding; |
| } else if (type.isDartCoreInt) { |
| return intEncoding; |
| } else if (type.isDartCoreDouble) { |
| return doubleEncoding; |
| } else if (type.isDartCoreString) { |
| return stringEncoding; |
| } |
| // Supertypes or type parameters are specialized in a consistent manner. |
| // TODO(ajcbik): inspect type structure semantically, not by display name |
| // and unify DartFuzz's DartType with analyzer DartType. |
| switch (type.asCode) { |
| case 'AbstractClassInstantiationError': |
| return abstractClassInstantiationErrorEncoding; |
| case 'ArgumentError': |
| return argumentErrorEncoding; |
| case 'AssertionError': |
| return assertionErrorEncoding; |
| case 'CastError': |
| return castErrorEncoding; |
| case 'ConcurrentModificationError': |
| return concurrentModificationErrorEncoding; |
| case 'CyclicInitializationError': |
| return cyclicInitializationErrorEncoding; |
| case 'Deprecated': |
| return deprecatedEncoding; |
| case 'E': |
| return intEncoding; |
| case 'Endian': |
| return endianEncoding; |
| case 'Error': |
| return errorEncoding; |
| case 'Exception': |
| return exceptionEncoding; |
| case 'Expando<double>': |
| return expandoDoubleEncoding; |
| case 'Expando<int>': |
| return expandoIntEncoding; |
| case 'FallThroughError': |
| return fallThroughErrorEncoding; |
| case 'Float32List': |
| return float32ListEncoding; |
| case 'Float32x4': |
| return float32x4Encoding; |
| case 'Float32x4List': |
| return float32x4ListEncoding; |
| case 'Float64List': |
| return float64ListEncoding; |
| case 'Float64x2': |
| return float64x2Encoding; |
| case 'Float64x2List': |
| return float64x2ListEncoding; |
| case 'FormatException': |
| return formatExceptionEncoding; |
| case 'IndexError': |
| return indexErrorEncoding; |
| case 'Int16List': |
| return int16ListEncoding; |
| case 'Int32List': |
| return int32ListEncoding; |
| case 'Int32x4': |
| return int32x4Encoding; |
| case 'Int32x4List': |
| return int32x4ListEncoding; |
| case 'Int64List': |
| return int64ListEncoding; |
| case 'Int8List': |
| return int8ListEncoding; |
| case 'IntegerDivisionByZeroException': |
| return integerDivisionByZeroExceptionEncoding; |
| case 'List': |
| case 'List<E>': |
| case 'List<Object>': |
| case 'List<dynamic>': |
| case 'List<int>': |
| return listIntEncoding; |
| case 'Map': |
| case 'Map<K, V>': |
| case 'Map<dynamic, dynamic>': |
| case 'Map<int, String>': |
| return mapIntStringEncoding; |
| case 'MapEntry': |
| case 'MapEntry<K, V>': |
| case 'MapEntry<dynamic, dynamic>': |
| case 'MapEntry<int, String>': |
| return mapEntryIntStringEncoding; |
| case 'Null': |
| return nullEncoding; |
| case 'NullThrownError': |
| return nullThrownErrorEncoding; |
| case 'Provisional': |
| return provisionalEncoding; |
| case 'RangeError': |
| return rangeErrorEncoding; |
| case 'RegExp': |
| return regExpEncoding; |
| case 'RuneIterator': |
| return runeIteratorEncoding; |
| case 'Runes': |
| return runesEncoding; |
| case 'Set': |
| case 'Set<E>': |
| case 'Set<Object>': |
| case 'Set<dynamic>': |
| case 'Set<int>': |
| return setIntEncoding; |
| case 'StackOverflowError': |
| return stackOverflowErrorEncoding; |
| case 'StateError': |
| return stateErrorEncoding; |
| case 'StringBuffer': |
| return stringBufferEncoding; |
| case 'Symbol': |
| return symbolEncoding; |
| case 'TypeError': |
| return typeErrorEncoding; |
| case 'Uint16List': |
| return uint16ListEncoding; |
| case 'Uint32List': |
| return uint32ListEncoding; |
| case 'Uint64List': |
| return uint64ListEncoding; |
| case 'Uint8ClampedList': |
| return uint8ClampedListEncoding; |
| case 'Uint8List': |
| return uint8ListEncoding; |
| case 'UnimplementedError': |
| return unimplementedErrorEncoding; |
| case 'UnsupportedError': |
| return unsupportedErrorEncoding; |
| case 'num': |
| return doubleEncoding; |
| case 'void': |
| return voidEncoding; |
| } |
| return '?'; |
| } |
| |
| List<String> protoString( |
| DartType? receiver, List<ParameterElement> parameters) { |
| final proto = [receiver == null ? voidEncoding : typeString(receiver)]; |
| // Construct prototype for non-named parameters. |
| for (var parameter in parameters) { |
| if (!parameter.isNamed) { |
| proto.add(typeString(parameter.type)); |
| } |
| } |
| // Use 'void' for an empty parameter list. |
| proto.length == 1 ? proto.add(voidEncoding) : proto; |
| return proto; |
| } |
| |
| List<DartLib> getTable(String ret) => typeToLibraryMethodsList.containsKey(ret) |
| ? typeToLibraryMethodsList[ret]! |
| : throw ArgumentError('Invalid ret value: $ret'); |
| |
| void addToTable(String ret, String name, List<String> proto, |
| {bool isMethod = true}) { |
| // If any of the type representations equal a question |
| // mark, this means that DartFuzz' type system cannot |
| // deal with such an expression yet. So drop the entry. |
| if (ret == '?' || proto.contains('?')) { |
| return; |
| } |
| // Avoid the exit function and other functions that give false divergences. |
| // Note: to prevent certain constructors from being emitted, update the |
| // exclude list in `shouldFilterConstructor` in gen_type_table.dart and |
| // regenerate the type table. |
| if (name == 'exit' || |
| name == 'pid' || |
| name == 'hashCode' || |
| name == 'exitCode' || |
| // TODO(fizaaluthra): Enable reciprocal and reciprocalSqrt after we resolve |
| // https://github.com/dart-lang/sdk/issues/39551 |
| name == 'reciprocal' || |
| name == 'reciprocalSqrt') { |
| return; |
| } |
| |
| List<Restriction>? restrictions; |
| // Restrict parameters for a few hardcoded cases, |
| // for example, to avoid excessive runtime or memory |
| // allocation in the generated fuzzing program. |
| if (name == 'padLeft' || name == 'padRight') { |
| for (var i = 0; i < proto.length - 1; ++i) { |
| if (proto[i] == intEncoding && proto[i + 1] == stringEncoding) { |
| restrictions = List<Restriction>.filled(proto.length, Restriction.none); |
| restrictions[i] = Restriction.small; |
| restrictions[i + 1] = Restriction.small; |
| break; |
| } |
| } |
| } else if (name == 'List<int>.filled') { |
| for (var i = 0; i < proto.length; ++i) { |
| if (proto[i] == intEncoding) { |
| restrictions = List<Restriction>.filled(proto.length, Restriction.none); |
| restrictions[i] = Restriction.small; |
| break; |
| } |
| } |
| } |
| // Add to table. |
| getTable(ret).add(DartLib(name, proto, restrictions, isMethod)); |
| // Every non-nullable result is also a valid nullable result. |
| if (!ret.endsWith("_NULLABLE")) { |
| getTable("${ret}_NULLABLE") |
| .add(DartLib(name, proto, restrictions, isMethod)); |
| } |
| } |
| |
| void dumpHeader() { |
| print(''' |
| // Copyright (c) 2019, the Dart project authors. Please see the AUTHORS file |
| // for details. All rights reserved. Use of this source code is governed by a |
| // BSD-style license that can be found in the LICENSE file. |
| |
| /// NOTE: this code has been generated automatically. |
| |
| import 'dartfuzz_type_table.dart'; |
| |
| /// Enum for different restrictions on parameters for library methods. |
| /// none - No restriction on the corresponding parameter. |
| /// small - Corresponding parameter should be a small value. |
| enum Restriction { |
| none, |
| small |
| } |
| |
| /// Class that represents Dart library methods. |
| /// |
| /// The invididual lists are organized by return type. |
| /// Proto is a list of DartTypes whose first element is the type of the |
| /// DartType of the receiver (DartType.VOID if none). The remaining elements are |
| /// DartTypes of the parameters. The second element is DartType.VOID if there |
| /// are no parameters. |
| class DartLib { |
| final String name; |
| final List<DartType> proto; |
| final List<Restriction>? restrictions; |
| final bool isMethod; |
| const DartLib(this.name, this.proto, this.isMethod, |
| {this.restrictions}); |
| Restriction getRestriction(int paramIndex) => (restrictions == null) ? |
| Restriction.none : restrictions![paramIndex]; |
| '''); |
| } |
| |
| void dumpTypeToLibraryMethodMap() { |
| print(' static final typeToLibraryMethods = {'); |
| for (var key in typeToLibraryMethodsListName.keys.toList()..sort()) { |
| if (typeToLibraryMethodsList[key]!.isNotEmpty) { |
| // Only output a mapping from type to library methods list name for those |
| // types that have a non-empty library methods list. |
| print(' $key: ${typeToLibraryMethodsListName[key]},'); |
| } |
| } |
| print(' };'); |
| } |
| |
| void dumpTypedDataFloatTypes() { |
| print(' static const typedDataFloatTypes = ['); |
| for (var type in typedDataFloatTypes) { |
| print(' $type,'); |
| } |
| print(' ];'); |
| } |
| |
| void dumpTable(String identifier, List<DartLib> table) { |
| print(' static const $identifier = ['); |
| table.sort((a, b) => (a.name.compareTo(b.name) == 0) |
| ? a.proto.join().compareTo(b.proto.join()) |
| : a.name.compareTo(b.name)); |
| table.forEach( |
| (t) => print(' DartLib(\'${t.name}\', ${t.proto}, ${t.isMethod}' |
| '${t.restrictions == null ? "" : ", " |
| "restrictions: ${t.restrictions}"}),')); |
| print(' ];'); |
| } |
| |
| void dumpFooter() { |
| print('}'); |
| } |