| #!/usr/bin/env python3 | 
 | # Copyright 2022 The Dawn 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. | 
 |  | 
 | import json, os, sys | 
 | from collections import namedtuple | 
 |  | 
 | from generator_lib import Generator, run_generator, FileRender | 
 |  | 
 |  | 
 | def parse_mask(mask): | 
 |     if mask: | 
 |         return int(mask, 0) | 
 |     return 0xffffffff | 
 |  | 
 |  | 
 | class Name: | 
 |     def __init__(self, name): | 
 |         self.name = name | 
 |         self.chunks = name.split(' ') | 
 |  | 
 |     def get(self): | 
 |         return self.name | 
 |  | 
 |     def CamelChunk(self, chunk): | 
 |         return chunk[0].upper() + chunk[1:] | 
 |  | 
 |     def canonical_case(self): | 
 |         return (' '.join(self.chunks)).lower() | 
 |  | 
 |     def concatcase(self): | 
 |         return ''.join(self.chunks) | 
 |  | 
 |     def camelCase(self): | 
 |         return self.chunks[0] + ''.join( | 
 |             [self.CamelChunk(chunk) for chunk in self.chunks[1:]]) | 
 |  | 
 |     def CamelCase(self): | 
 |         return ''.join([self.CamelChunk(chunk) for chunk in self.chunks]) | 
 |  | 
 |     def SNAKE_CASE(self): | 
 |         return '_'.join([chunk.upper() for chunk in self.chunks]) | 
 |  | 
 |     def snake_case(self): | 
 |         return '_'.join(self.chunks) | 
 |  | 
 |     def js_enum_case(self): | 
 |         result = self.chunks[0].lower() | 
 |         for chunk in self.chunks[1:]: | 
 |             if not result[-1].isdigit(): | 
 |                 result += '-' | 
 |             result += chunk.lower() | 
 |         return result | 
 |  | 
 |  | 
 | class Architecture: | 
 |     def __init__(self, name, json_data, mask): | 
 |         self.name = Name(name) | 
 |         self.devices = [] | 
 |  | 
 |         mask_num = parse_mask(mask) | 
 |  | 
 |         for device in json_data: | 
 |             device_num = int(device, 0) | 
 |  | 
 |             # Don't allow duplicate entries | 
 |             assert device not in self.devices, 'Architecture "{}" contained duplicate deviceID "{}"'.format( | 
 |                 self.name.get(), device) | 
 |             # Ensure that all device IDs don't contain bits outside the mask | 
 |             assert device_num & mask_num == device_num, 'Architecture "{}" contained deviceID "{}" which doesn\'t match the given mask of "{}"'.format( | 
 |                 self.name.get(), device, mask) | 
 |  | 
 |             self.devices.append(device) | 
 |  | 
 |  | 
 | class DeviceSet: | 
 |     def __init__(self, json_data): | 
 |         self.mask = None | 
 |         self.internal = False | 
 |  | 
 |         if 'mask' in json_data: | 
 |             self.mask = json_data['mask'] | 
 |  | 
 |         if 'internal' in json_data: | 
 |             self.internal = json_data['internal'] | 
 |  | 
 |         self.architectures = [] | 
 |         if 'architecture' in json_data: | 
 |             for (arch_name, arch_data) in json_data['architecture'].items(): | 
 |                 # Skip any entries that start with an underscore. Used for comments. | 
 |                 if arch_name[0] == '_': | 
 |                     continue | 
 |  | 
 |                 architecture = Architecture(arch_name, arch_data, self.mask) | 
 |  | 
 |                 # Validate that deviceIDs are only allowed to be in one Architecture at a time | 
 |                 for other_architecture in self.architectures: | 
 |                     for device in architecture.devices: | 
 |                         assert device not in other_architecture.devices, 'Architectures "{}" and "{}" both contain deviceID "{}"'.format( | 
 |                             architecture.name.get(), | 
 |                             other_architecture.name.get(), device) | 
 |  | 
 |                 self.architectures.append(architecture) | 
 |  | 
 |     def validate_devices(self, other_devices, other_mask): | 
 |         combined_mask = parse_mask(self.mask) & parse_mask(other_mask) | 
 |  | 
 |         for other_device in other_devices: | 
 |             other_device_num = int(other_device, 0) & combined_mask | 
 |             for architecture in self.architectures: | 
 |                 for device in architecture.devices: | 
 |                     device_num = int(device, 0) & combined_mask | 
 |                     assert device_num != other_device_num, 'DeviceID "{}" & mask "{}" conflicts with deviceId "{}" & mask "{}" in architecture "{}"'.format( | 
 |                         other_device, other_mask, device, self.mask, | 
 |                         architecture.name.get()) | 
 |  | 
 |     def maskDeviceId(self): | 
 |         if not self.mask: | 
 |             return '' | 
 |         return ' & ' + self.mask | 
 |  | 
 |  | 
 | class Vendor: | 
 |     def __init__(self, name, json_data): | 
 |         self.name = Name(name) | 
 |         self.id = json_data['id'] | 
 |  | 
 |         architecture_dict = {} | 
 |         internal_architecture_dict = {} | 
 |  | 
 |         self.device_sets = [] | 
 |         if 'devices' in json_data: | 
 |             for device_data in json_data['devices']: | 
 |                 device_set = DeviceSet(device_data) | 
 |  | 
 |                 for architecture in device_set.architectures: | 
 |  | 
 |                     # Validate that deviceIDs are unique across device sets | 
 |                     for other_device_set in self.device_sets: | 
 |                         # Only validate device IDs between internal and public device sets. | 
 |                         if other_device_set.internal == device_set.internal: | 
 |                             assert device_set.mask != other_device_set.mask, 'Vendor "{}" contained duplicate device masks "{}"'.format( | 
 |                                 self.name.get(), device_set.mask) | 
 |                             other_device_set.validate_devices( | 
 |                                 architecture.devices, device_set.mask) | 
 |  | 
 |                         # Validate that architecture names are unique between internal and public device sets. | 
 |                         else: | 
 |                             for other_architecture in other_device_set.architectures: | 
 |                                 assert architecture.name.canonical_case( | 
 |                                 ) != other_architecture.name.canonical_case( | 
 |                                 ), '"{}" is defined as both an internal and public architecture'.format( | 
 |                                     architecture.name.get()) | 
 |  | 
 |                     if device_set.internal: | 
 |                         internal_architecture_dict[ | 
 |                             architecture.name.canonical_case( | 
 |                             )] = architecture.name | 
 |                     else: | 
 |                         architecture_dict[architecture.name.canonical_case( | 
 |                         )] = architecture.name | 
 |  | 
 |                 self.device_sets.append(device_set) | 
 |  | 
 |         # List of unique architecture names under this vendor | 
 |         self.architecture_names = architecture_dict.values() | 
 |         self.internal_architecture_names = internal_architecture_dict.values() | 
 |  | 
 |  | 
 | def parse_json(json): | 
 |     vendors = [] | 
 |     internal_architecture_count = 0 | 
 |  | 
 |     for (vendor_name, vendor_data) in json['vendors'].items(): | 
 |         # Skip vendors that have a leading underscore. Those are intended to be "comments". | 
 |         if vendor_name[0] == '_': | 
 |             continue | 
 |  | 
 |         vendor = Vendor(vendor_name, vendor_data) | 
 |         vendors.append(vendor) | 
 |         internal_architecture_count += len(vendor.internal_architecture_names) | 
 |  | 
 |     return { | 
 |         'vendors': vendors, | 
 |         'has_internal': internal_architecture_count > 0 | 
 |     } | 
 |  | 
 |  | 
 | class DawnGpuInfoGenerator(Generator): | 
 |     def get_description(self): | 
 |         return "Generates GPU Info Dawn code." | 
 |  | 
 |     def add_commandline_arguments(self, parser): | 
 |         parser.add_argument('--gpu-info-json', | 
 |                             required=True, | 
 |                             type=str, | 
 |                             help='The GPU Info JSON definition to use.') | 
 |  | 
 |     def get_dependencies(self, args): | 
 |         return [os.path.abspath(args.gpu_info_json)] | 
 |  | 
 |     def get_file_renders(self, args): | 
 |         with open(args.gpu_info_json) as f: | 
 |             loaded_json = json.loads(f.read()) | 
 |  | 
 |         params = parse_json(loaded_json) | 
 |  | 
 |         return [ | 
 |             FileRender("dawn/common/GPUInfo.h", | 
 |                        "src/dawn/common/GPUInfo_autogen.h", [params]), | 
 |             FileRender("dawn/common/GPUInfo.cpp", | 
 |                        "src/dawn/common/GPUInfo_autogen.cpp", [params]), | 
 |         ] | 
 |  | 
 |  | 
 | if __name__ == "__main__": | 
 |     sys.exit(run_generator(DawnGpuInfoGenerator())) |