+++ /dev/null
-#
-# Copyright 2017 Advanced Micro Devices, Inc.
-#
-# Permission is hereby granted, free of charge, to any person obtaining a
-# copy of this software and associated documentation files (the "Software"),
-# to deal in the Software without restriction, including without limitation
-# on the rights to use, copy, modify, merge, publish, distribute, sub
-# license, and/or sell copies of the Software, and to permit persons to whom
-# the Software is furnished to do so, subject to the following conditions:
-#
-# The above copyright notice and this permission notice (including the next
-# paragraph) shall be included in all copies or substantial portions of the
-# Software.
-#
-# THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
-# IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
-# FITNESS FOR A PARTICULAR PURPOSE AND NON-INFRINGEMENT. IN NO EVENT SHALL
-# THE AUTHOR(S) AND/OR THEIR SUPPLIERS BE LIABLE FOR ANY CLAIM,
-# DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR
-# OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE
-# USE OR OTHER DEALINGS IN THE SOFTWARE.
-#
-"""
-Script that generates the mapping from Vulkan VK_FORMAT_xxx to gfx10
-IMG_FORMAT_xxx enums.
-"""
-
-from __future__ import absolute_import, division, print_function, unicode_literals
-
-import json
-import mako.template
-import os
-import re
-import sys
-
-AMD_REGISTERS = os.path.abspath(os.path.join(os.path.dirname(sys.argv[0]), "../registers"))
-sys.path.extend([AMD_REGISTERS])
-
-from regdb import Object, RegisterDatabase
-from vk_format_parse import *
-
-# ----------------------------------------------------------------------------
-# Hard-coded mappings
-
-def hardcoded_format(hw_enum):
- return Object(img_format=hw_enum, flags=[])
-
-HARDCODED = {
- 'VK_FORMAT_E5B9G9R9_UFLOAT_PACK32': hardcoded_format('5_9_9_9_FLOAT'),
- 'VK_FORMAT_B10G11R11_UFLOAT_PACK32': hardcoded_format('10_11_11_FLOAT'), # NOTE: full set of int/unorm/etc. exists
-
- # BC
- 'VK_FORMAT_BC1_RGB_UNORM_BLOCK': hardcoded_format('BC1_UNORM'),
- 'VK_FORMAT_BC1_RGBA_UNORM_BLOCK': hardcoded_format('BC1_UNORM'),
- 'VK_FORMAT_BC1_RGB_SRGB_BLOCK': hardcoded_format('BC1_SRGB'),
- 'VK_FORMAT_BC1_RGBA_SRGB_BLOCK': hardcoded_format('BC1_SRGB'),
- 'VK_FORMAT_BC2_UNORM_BLOCK': hardcoded_format('BC2_UNORM'),
- 'VK_FORMAT_BC2_SRGB_BLOCK': hardcoded_format('BC2_SRGB'),
- 'VK_FORMAT_BC3_UNORM_BLOCK': hardcoded_format('BC3_UNORM'),
- 'VK_FORMAT_BC3_SRGB_BLOCK': hardcoded_format('BC3_SRGB'),
- 'VK_FORMAT_BC4_UNORM_BLOCK': hardcoded_format('BC4_UNORM'),
- 'VK_FORMAT_BC4_SNORM_BLOCK': hardcoded_format('BC4_SNORM'),
- 'VK_FORMAT_BC5_UNORM_BLOCK': hardcoded_format('BC5_UNORM'),
- 'VK_FORMAT_BC5_SNORM_BLOCK': hardcoded_format('BC5_SNORM'),
- 'VK_FORMAT_BC6H_UFLOAT_BLOCK': hardcoded_format('BC6_UFLOAT'),
- 'VK_FORMAT_BC6H_SFLOAT_BLOCK': hardcoded_format('BC6_SFLOAT'),
- 'VK_FORMAT_BC7_UNORM_BLOCK': hardcoded_format('BC7_UNORM'),
- 'VK_FORMAT_BC7_SRGB_BLOCK': hardcoded_format('BC7_SRGB'),
-
- # DS
- 'VK_FORMAT_D16_UNORM_S8_UINT': hardcoded_format('INVALID'),
- 'VK_FORMAT_D24_UNORM_S8_UINT': hardcoded_format('8_24_UNORM'),
- 'VK_FORMAT_D32_SFLOAT_S8_UINT': hardcoded_format('X24_8_32_FLOAT'),
-}
-
-
-# ----------------------------------------------------------------------------
-# Main script
-
-header_template = mako.template.Template("""\
-// DO NOT EDIT -- AUTOMATICALLY GENERATED
-
-#define FMT(_img_format, ...) \
- { .img_format = V_008F0C_IMG_FORMAT_##_img_format, \
- ##__VA_ARGS__ }
-
-% for vk_format, args in formats:
- % if args is not None:
- static const struct gfx10_format gfx10_info_${vk_format} = FMT(${args});
- % else:
- /* ${vk_format} is not supported */
- % endif
-% endfor
-
-static const struct gfx10_format gfx10_unsupported_format = { 0 };
-
-static inline const struct gfx10_format* gfx10_format_description(VkFormat format)
-{
- switch(format)
- {
- % for vk_format, args in formats:
- % if args is not None:
- case ${vk_format}: return &gfx10_info_${vk_format};
- % endif
- % endfor
- default: return &gfx10_unsupported_format;
- }
-}
-""")
-
-class Gfx10Format(object):
- RE_plain_channel = re.compile(r'X?([0-9]+)')
-
- def __init__(self, enum_entry):
- self.img_format = enum_entry.name[11:]
- self.flags = getattr(enum_entry, 'flags', [])
-
- code = self.img_format.split('_')
-
- self.plain_chan_sizes = []
- for i, chan_code in enumerate(code):
- m = self.RE_plain_channel.match(chan_code)
- if m is None:
- break
- self.plain_chan_sizes.append(int(m.group(1)))
- # Keep the bit sizes in little-endian order
- self.plain_chan_sizes.reverse()
-
- self.code = code[i:]
-
-
-class Gfx10FormatMapping(object):
- def __init__(self, vk_formats, gfx10_formats):
- self.vk_formats = vk_formats
- self.gfx10_formats = gfx10_formats
-
- self.plain_gfx10_formats = dict(
- (tuple(['_'.join(fmt.code)] + fmt.plain_chan_sizes), fmt)
- for fmt in gfx10_formats if fmt.plain_chan_sizes
- )
-
- def map(self, fmt):
- if fmt.layout == PLAIN:
- chan_type = set([chan.type for chan in fmt.le_channels if chan.type != VOID])
- chan_norm = set([chan.norm for chan in fmt.le_channels if chan.type != VOID])
- chan_pure = set([chan.pure for chan in fmt.le_channels if chan.type != VOID])
- if len(chan_type) > 1 or len(chan_norm) > 1 or len(chan_pure) > 1:
- print(('Format {fmt.name} has inconsistent channel types: ' +
- '{chan_type} {chan_norm} {chan_pure}')
- .format(**locals()),
- file=sys.stderr)
- return None
-
- chan_type = chan_type.pop()
- chan_norm = chan_norm.pop()
- chan_pure = chan_pure.pop()
- chan_sizes = [chan.size for chan in fmt.le_channels if chan.size != 0]
-
- extra_flags = []
-
- if fmt.colorspace == SRGB:
- assert chan_type == UNSIGNED and chan_norm
- num_format = 'SRGB'
- else:
- if chan_type == UNSIGNED:
- if chan_pure:
- num_format = 'UINT'
- elif chan_sizes[0] == 32:
- # Shader-based work-around for 32-bit non-pure-integer
- num_format = 'UINT'
- extra_flags.append('buffers_only')
- elif chan_norm:
- num_format = 'UNORM'
- else:
- num_format = 'USCALED'
- elif chan_type == SIGNED:
- if chan_pure:
- num_format = 'SINT'
- elif chan_sizes[0] == 32:
- # Shader-based work-around for 32-bit non-pure-integer
- num_format = 'SINT'
- extra_flags.append('buffers_only')
- elif chan_norm:
- num_format = 'SNORM'
- else:
- num_format = 'SSCALED'
- elif chan_type == FLOAT:
- num_format = 'FLOAT'
-
- if chan_sizes[0] == 64:
- # Shader-based work-around for doubles
- if len(chan_sizes) % 2 == 1:
- # 1 or 3 loads for 1 or 3 double channels
- chan_sizes = [32, 32]
- else:
- # 1 or 2 loads for 2 or 4 double channels
- chan_sizes = [32, 32, 32, 32]
- extra_flags.append('buffers_only')
- else:
- # Shader-based work-around
- assert chan_type == FIXED
- assert chan_sizes[0] == 32
- num_format = 'SINT'
- extra_flags.append('buffers_only')
-
- # These are not supported as render targets, so we don't support
- # them as images either.
- if (len(chan_sizes) == 3 and chan_sizes[0] in (8, 16, 32) and
- chan_sizes[0] == chan_sizes[1]):
- extra_flags.append('buffers_only')
- if chan_sizes[0] in (8, 16):
- # Shader-based work-around: one load per channel
- chan_sizes = [chan_sizes[0]]
-
- # Don't expose SRGB buffer formats
- if 'buffers_only' in extra_flags and fmt.colorspace == SRGB:
- return None
-
- # Don't support 4_4 because it's not supported as render targets
- # and it's useless in other cases.
- if len(chan_sizes) == 2 and chan_sizes[0] == 4:
- return None
-
- key = tuple([num_format] + chan_sizes)
- if key not in self.plain_gfx10_formats:
- return None
-
- gfx10_fmt = self.plain_gfx10_formats[key]
- return Object(
- img_format=gfx10_fmt.img_format,
- flags=gfx10_fmt.flags + extra_flags,
- )
-
- return None
-
-
-if __name__ == '__main__':
- vk_formats = parse(sys.argv[1])
-
- with open(sys.argv[2], 'r') as filp:
- db = RegisterDatabase.from_json(json.load(filp))
-
- gfx10_formats = [Gfx10Format(entry) for entry in db.enum('IMG_FORMAT').entries]
-
- mapping = Gfx10FormatMapping(vk_formats, gfx10_formats)
-
- formats = []
- for fmt in vk_formats:
- if fmt.name in HARDCODED:
- obj = HARDCODED[fmt.name]
- else:
- obj = mapping.map(fmt)
-
- if obj is not None:
- args = obj.img_format
- if 'buffers_only' in obj.flags:
- args += ', .buffers_only = 1'
- else:
- args = None
- formats.append((fmt.name, args))
-
- print(header_template.render(formats=formats))