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, write
8from spec_tools.attributes import ExternSyncEntry
9from spec_tools.util import getElemName
10
11import pdb
12
13class FormatsOutputGenerator(OutputGenerator):
14    """FormatsOutputGenerator - subclass of OutputGenerator.
15    Generates AsciiDoc includes of the table for the format chapters
16    of the API specification.
17
18    ---- methods ----
19    FormatsOutputGenerator(errFile, warnFile, diagFile) - args as for
20      OutputGenerator. Defines additional internal state.
21    ---- methods overriding base class ----
22    genCmd(cmdinfo)"""
23
24    def __init__(self, *args, **kwargs):
25        super().__init__(*args, **kwargs)
26
27    def beginFile(self, genOpts):
28        OutputGenerator.beginFile(self, genOpts)
29
30        # List of all the formats elements
31        self.formats = []
32        # <format, condition as asciidoc string>
33        self.format_conditions = dict()
34        # <class, {'formats' : [], 'meta' : {} }>
35        self.format_classes = dict()
36        # {'packedSize' : ['format', 'format', ...]}
37        self.packed_info = dict()
38        # {VkFormat : SpirvFormat}
39        self.spirv_image_format = dict()
40        # <format, [{plane_info}, ...]>
41        self.plane_format = dict()
42
43    def endFile(self):
44
45        # Generate compatibility table
46        compatibility_table = []
47        for class_name, info in self.format_classes.items():
48            # Do an initial loop of formats in class to see if whole class is a single condition
49            class_condition = None
50            for index, format in enumerate(info['formats']):
51                condition = self.format_conditions[format]
52                if (condition == None) or (class_condition != None and class_condition != condition):
53                    class_condition = None
54                    break
55                else:
56                    class_condition = condition
57
58            # If not single class condition for the class, next check if a single format has a condition
59            # Move all condition formats to the front of array to make listing the formats in table
60            if class_condition == None:
61                condition_list = []
62                noncondition_list = []
63                for index, format in enumerate(info['formats']):
64                    if self.format_conditions[format] == None:
65                        noncondition_list.append(format)
66                    else:
67                        condition_list.append(format)
68                info['formats'] = condition_list + noncondition_list
69
70            if class_condition != None:
71                compatibility_table.append('ifdef::{}[]'.format(class_condition))
72
73            compatibility_table.append("| {} +".format(class_name))
74            compatibility_table.append("  Block size {} byte +".format(info['meta']['blockSize']))
75            compatibility_table.append("  {} block extent +".format(info['meta']['blockExtent'].replace(",", "x")))
76            compatibility_table.append("  {} texel/block |".format(info['meta']['texelsPerBlock']))
77
78            for index, format in enumerate(info['formats']):
79                format_condition = self.format_conditions[format]
80                if format_condition != None and class_condition == None:
81                    compatibility_table.append('ifdef::{}[]'.format(format_condition))
82                suffix = ", +" if index != len(info['formats']) - 1 else ""
83                compatibility_table.append("                    ename:{}{}".format(format, suffix))
84                if format_condition != None and class_condition == None:
85                    compatibility_table.append('endif::{}[]'.format(format_condition))
86
87            if class_condition != None:
88                compatibility_table.append('endif::{}[]'.format(class_condition))
89        self.writeBlock(f'compatibility{self.file_suffix}', compatibility_table)
90
91        # Generate packed format list
92        packed_table = []
93        for packed_size, formats in self.packed_info.items():
94            packed_table.append('  * <<formats-packed-{}-bit,Packed into {}-bit data types>>:'.format(packed_size, packed_size))
95            # Do an initial loop of formats with same packed size to group conditional together for easier reading of final asciidoc
96            sorted_formats = dict() # {condition : formats}
97            for format in formats:
98                format_condition = self.format_conditions[format]
99                if format_condition == None:
100                    format_condition = "None" # to allow as a key in the dict
101                if format_condition not in sorted_formats:
102                    sorted_formats[format_condition] = []
103                sorted_formats[format_condition].append(format)
104
105            for condition, condition_formats in sorted_formats.items():
106                if condition != "None":
107                    packed_table.append('ifdef::{}[]'.format(condition))
108                for format in condition_formats:
109                    packed_table.append('  ** ename:{}'.format(format))
110                if condition != "None":
111                    packed_table.append('endif::{}[]'.format(condition))
112        self.writeBlock(f'packed{self.file_suffix}', packed_table)
113
114        # Generate SPIR-V Image Format Compatibility
115        spirv_image_format_table = []
116        spirv_image_format_table.append('|code:Unknown|Any')
117        for vk_format, spirv_format in self.spirv_image_format.items():
118            spirv_image_format_table.append('|code:{}|ename:{}'.format(spirv_format, vk_format))
119        self.writeBlock(f'spirvimageformat{self.file_suffix}', spirv_image_format_table)
120
121        # Generate Plane Format Compatibility Table
122        plane_format_table = []
123        for format_name, plane_infos in self.plane_format.items():
124            format_condition = self.format_conditions[format_name]
125            # The table is already in a ifdef::VK_VERSION_1_1,VK_KHR_sampler_ycbcr_conversion[]
126            # so no need to duplicate the condition
127            add_condition = False if format_condition == 'None' or format_condition == 'VK_VERSION_1_1,VK_KHR_sampler_ycbcr_conversion' else True
128
129            if add_condition:
130                plane_format_table.append('ifdef::{}[]'.format(format_condition))
131
132            plane_format_table.append('4+| *ename:{}*'.format(format_name))
133            for plane_info in plane_infos:
134                width_divisor = 'w'
135                height_divisor = 'h'
136                if plane_info['widthDivisor'] != 1:
137                    width_divisor += '/{}'.format(plane_info['widthDivisor'])
138                if plane_info['heightDivisor'] != 1:
139                    height_divisor += '/{}'.format(plane_info['heightDivisor'])
140
141                plane_format_table.append('^| {} ^| ename:{} ^| {} ^| {}'.format(plane_info['index'],
142                                                                                 plane_info['compatible'],
143                                                                                 width_divisor,
144                                                                                 height_divisor))
145            if add_condition:
146                plane_format_table.append('endif::{}[]'.format(format_condition))
147        self.writeBlock(f'planeformat{self.file_suffix}', plane_format_table)
148
149        # Finish processing in superclass
150        OutputGenerator.endFile(self)
151
152    def writeBlock(self, basename, contents):
153        """Generate an include file.
154
155        - directory - subdirectory to put file in
156        - basename - base name of the file
157        - contents - contents of the file (Asciidoc boilerplate aside)"""
158
159        filename = self.genOpts.directory + '/' + basename
160        self.logMsg('diag', '# Generating include file:', filename)
161        with open(filename, 'w', encoding='utf-8') as fp:
162            write(self.genOpts.conventions.warning_comment, file=fp)
163
164            if len(contents) > 0:
165                for str in contents:
166                    write(str, file=fp)
167            else:
168                self.logMsg('diag', '# No contents for:', filename)
169
170    def genFormat(self, format, formatinfo, alias):
171        """Generate Formats
172
173        formatinfo - dictionary entry for an XML <format> element
174        name - name attribute of format.elem"""
175
176        OutputGenerator.genFormat(self, format, formatinfo, alias)
177        elem = format.elem
178        format_name = elem.get('name')
179
180        self.formats.append(elem)
181        self.format_conditions[format_name] = format.condition
182
183        # Create format class data structure to be processed later
184        class_name = elem.get('class')
185        class_meta = {
186            'blockSize' : elem.get('blockSize'),
187            'texelsPerBlock' : elem.get('texelsPerBlock'),
188            # default extent
189            'blockExtent' : "1,1,1" if elem.get('blockExtent') == None else elem.get('blockExtent')
190        }
191
192        if class_name in self.format_classes:
193            self.format_classes[class_name]['formats'].append(format_name)
194            # Assert all classes are using same meta info
195            if class_meta != self.format_classes[class_name]['meta']:
196                self.logMsg('error', 'Class meta info is not consistent for class ', class_name)
197        else:
198            self.format_classes[class_name] = {
199                'formats' : [format_name],
200                'meta' : class_meta
201            }
202
203        # Build list of formats with packed info in xml
204        packed = elem.get('packed')
205        if packed is not None:
206            if packed not in self.packed_info:
207                self.packed_info[packed] = []
208            self.packed_info[packed].append(format_name)
209
210        # Currently there is only at most one <spirvimageformat>
211        spirv_image_format = elem.find('spirvimageformat')
212        if (spirv_image_format is not None):
213            self.spirv_image_format[format_name] = spirv_image_format.get('name')
214
215        for plane in elem.iterfind('plane'):
216            if format_name not in self.plane_format:
217                # create list if first time
218                self.plane_format[format_name] = []
219            self.plane_format[format_name].append({
220                'index' : int(plane.get('index')),
221                'widthDivisor' : int(plane.get('widthDivisor')),
222                'heightDivisor' : int(plane.get('heightDivisor')),
223                'compatible' : plane.get('compatible'),
224            })
225