1#!/usr/bin/python3 -i 2# 3# Copyright 2013-2023 The Khronos Group Inc. 4# 5# SPDX-License-Identifier: Apache-2.0 6 7from generator import OutputGenerator, enquote, write 8from scriptgenerator import ScriptOutputGenerator 9 10def nilquote(s): 11 if s: 12 return enquote(s) 13 else: 14 return 'nil' 15 16def makeHash(name): 17 return '@{} = {{'.format(name) 18 19class RubyOutputGenerator(ScriptOutputGenerator): 20 """RubyOutputGenerator - subclass of ScriptOutputGenerator. 21 Generates Ruby data structures describing API names and 22 relationships.""" 23 24 def __init__(self, *args, **kwargs): 25 super().__init__(*args, **kwargs) 26 27 def beginDict(self, name): 28 """String starting definition of a named dictionary""" 29 return f'@{name} = {{' 30 31 def endDict(self): 32 """ String ending definition of a named dictionary""" 33 return '}' 34 35 def writeDict(self, dict, name, printValues = True): 36 """Write dictionary as a Ruby hash with the given name. 37 If printValues is False, just output keys with nil values.""" 38 39 write(self.beginDict(name), file=self.outFile) 40 for key in sorted(dict): 41 if printValues: 42 value = nilquote(dict[key]) 43 else: 44 value = 'nil' 45 write(f'{enquote(key)} => {value},', file=self.outFile) 46 write(self.endDict(), file=self.outFile) 47 48 def writeList(self, l, name): 49 """Write list l as a Ruby hash with the given name""" 50 51 self.writeDict(l, name, printValues = False) 52 53 def makeAccessor(self, name): 54 """Create an accessor method for the hash 'name'""" 55 write('def {}'.format(name), file=self.outFile) 56 write(' @{}'.format(name), file=self.outFile) 57 write('end', file=self.outFile) 58 59 def endFile(self): 60 # Creates the inverse mapping of nonexistent APIs to their aliases. 61 super().createInverseMap() 62 63 # Print out all the dictionaries as Ruby strings. 64 # Use a simple container class for namespace control 65 write('class APInames\n', ' def initialize', file=self.outFile) 66 67 dicts = ( [ self.basetypes, 'basetypes' ], 68 [ self.consts, 'consts' ], 69 [ self.enums, 'enums' ], 70 [ self.flags, 'flags' ], 71 [ self.funcpointers, 'funcpointers' ], 72 [ self.protos, 'protos' ], 73 [ self.structs, 'structs' ], 74 [ self.handles, 'handles' ], 75 [ self.defines, 'defines' ], 76 [ self.typeCategory, 'typeCategory' ], 77 [ self.alias, 'aliases' ], 78 [ self.nonexistent, 'nonexistent' ], 79 ) 80 for (dict, name) in dicts: 81 self.writeDict(dict, name) 82 83 # Dictionary containing the relationships of a type 84 # (e.g. a dictionary with each related type as keys). 85 write(self.beginDict('mapDict'), file=self.outFile) 86 for baseType in sorted(self.mapDict): 87 # Not actually including the relationships yet 88 write('{} => {},'.format(enquote(baseType), 'nil'), 89 file=self.outFile) 90 write(self.endDict(), file=self.outFile) 91 92 # List of included feature names 93 self.writeList(sorted(self.features), 'features') 94 95 # Generate feature <-> interface mappings 96 for feature in self.features: 97 self.mapInterfaces(feature) 98 99 # Write out the reverse map from APIs to requiring features 100 write(self.beginDict('requiredBy'), file=self.outFile) 101 for api in sorted(self.apimap): 102 # Sort requirements by first feature in each one 103 deps = sorted(self.apimap[api], key = lambda dep: dep[0]) 104 reqs = ', '.join('[{}, {}]'.format(nilquote(dep[0]), nilquote(dep[1])) for dep in deps) 105 write('{} => [{}],'.format(enquote(api), reqs), file=self.outFile) 106 write(self.endDict(), file=self.outFile) 107 108 # Remainder of the class definition 109 # End initialize method 110 write('end', file=self.outFile) 111 112 # Accessor methods 113 for (_, name) in dicts: 114 self.makeAccessor(name) 115 self.makeAccessor('features') 116 117 # Class end 118 write('end', file=self.outFile) 119 120 super().endFile() 121