Vulkan SDK for Android 1.1.1 Mali Developer Center
All Classes Functions Variables Enumerations Enumerator Pages
assets.cpp
1 /* Copyright (c) 2016-2017, ARM Limited and Contributors
2  *
3  * SPDX-License-Identifier: MIT
4  *
5  * Permission is hereby granted, free of charge,
6  * to any person obtaining a copy of this software and associated documentation files (the "Software"),
7  * to deal in the Software without restriction, including without limitation the rights to
8  * use, copy, modify, merge, publish, distribute, sublicense, and/or sell copies of the Software,
9  * and to permit persons to whom the Software is furnished to do so, subject to the following conditions:
10  *
11  * The above copyright notice and this permission notice shall be included in all copies or substantial portions of the Software.
12  *
13  * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR IMPLIED,
14  * INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
15  * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT.
16  * IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY,
17  * WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
18  * OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
19  */
20 
21 #include "assets.hpp"
22 #include "common.hpp"
23 #include "platform/os.hpp"
24 #include "platform/platform.hpp"
25 #include <stdio.h>
26 #include <vector>
27 
28 #define STB_IMAGE_STATIC
29 #define STB_IMAGE_IMPLEMENTATION
30 #include <stb/stb_image.h>
31 
32 using namespace std;
33 
34 namespace MaliSDK
35 {
36 
37 VkShaderModule loadShaderModule(VkDevice device, const char *pPath)
38 {
39  vector<uint32_t> buffer;
40  if (FAILED(OS::getAssetManager().readBinaryFile(&buffer, pPath)))
41  {
42  LOGE("Failed to read SPIR-V file: %s.\n", pPath);
43  return VK_NULL_HANDLE;
44  }
45 
46  VkShaderModuleCreateInfo moduleInfo = { VK_STRUCTURE_TYPE_SHADER_MODULE_CREATE_INFO };
47  moduleInfo.codeSize = buffer.size() * sizeof(uint32_t);
48  moduleInfo.pCode = buffer.data();
49 
50  VkShaderModule shaderModule;
51  VK_CHECK(vkCreateShaderModule(device, &moduleInfo, nullptr, &shaderModule));
52  return shaderModule;
53 }
54 
55 Result loadRgba8888TextureFromAsset(const char *pPath, vector<uint8_t> *pBuffer, unsigned *pWidth, unsigned *pHeight)
56 {
57  vector<uint8_t> compressed;
58  if (FAILED(OS::getAssetManager().readBinaryFile(&compressed, pPath)))
59  {
60  LOGE("Failed to read texture: %s.\n", pPath);
61  return RESULT_ERROR_IO;
62  }
63 
64  int x, y, comp;
65 
66  uint8_t *pResult = stbi_load_from_memory(compressed.data(), compressed.size(), &x, &y, &comp, STBI_rgb_alpha);
67  if (!pResult || comp != STBI_rgb_alpha)
68  {
69  LOGE("Failed to decompress texture: %s.\n", pPath);
70  free(pResult);
71  return RESULT_ERROR_GENERIC;
72  }
73 
74  pBuffer->clear();
75  pBuffer->insert(end(*pBuffer), pResult, pResult + x * y * 4);
76  *pWidth = x;
77  *pHeight = y;
78  free(pResult);
79 
80  return RESULT_SUCCESS;
81 }
82 
84 struct ASTCHeader
85 {
87  uint8_t magic[4];
89  uint8_t blockdimX;
91  uint8_t blockdimY;
93  uint8_t blockdimZ;
95  uint8_t xsize[3];
97  uint8_t ysize[3];
99  uint8_t zsize[3];
100 };
101 static_assert(sizeof(ASTCHeader) == 16, "Packed ASTC header struct is not 16 bytes.");
102 
103 #define ASTC_MAGIC 0x5CA1AB13
104 
105 Result loadASTCTextureFromAsset(const char *pPath, vector<uint8_t> *pBuffer, unsigned *pWidth, unsigned *pHeight,
106  VkFormat *pFormat)
107 {
108  vector<uint8_t> compressed;
109  if (FAILED(OS::getAssetManager().readBinaryFile(&compressed, pPath)))
110  {
111  LOGE("Failed to read ASTC texture: %s.\n", pPath);
112  return RESULT_ERROR_IO;
113  }
114 
115  if (compressed.size() < sizeof(ASTCHeader))
116  return RESULT_ERROR_GENERIC;
117 
118  ASTCHeader header;
119  memcpy(&header, compressed.data(), sizeof(ASTCHeader));
120  uint32_t magic = header.magic[0] | (uint32_t(header.magic[1]) << 8) | (uint32_t(header.magic[2]) << 16) |
121  (uint32_t(header.magic[3]) << 24);
122 
123  if (magic != ASTC_MAGIC)
124  {
125  LOGE("Texture %s is not ASTC.\n", pPath);
126  return RESULT_ERROR_GENERIC;
127  }
128 
129  if (header.blockdimZ != 1)
130  {
131  LOGE("ASTC 3D textures not supported yet in Vulkan.\n");
132  return RESULT_ERROR_GENERIC;
133  }
134 
135  if (header.blockdimX == 4 && header.blockdimY == 4) // 4x4
136  *pFormat = VK_FORMAT_ASTC_4x4_UNORM_BLOCK;
137  else if (header.blockdimX == 5 && header.blockdimY == 4) // 5x4
138  *pFormat = VK_FORMAT_ASTC_5x4_UNORM_BLOCK;
139  else if (header.blockdimX == 6 && header.blockdimY == 5) // 6x5
140  *pFormat = VK_FORMAT_ASTC_6x5_UNORM_BLOCK;
141  else if (header.blockdimX == 6 && header.blockdimY == 6) // 6x6
142  *pFormat = VK_FORMAT_ASTC_6x6_UNORM_BLOCK;
143  else if (header.blockdimX == 8 && header.blockdimY == 5) // 8x5
144  *pFormat = VK_FORMAT_ASTC_8x5_UNORM_BLOCK;
145  else if (header.blockdimX == 8 && header.blockdimY == 6) // 8x6
146  *pFormat = VK_FORMAT_ASTC_8x6_UNORM_BLOCK;
147  else if (header.blockdimX == 8 && header.blockdimY == 8) // 8x8
148  *pFormat = VK_FORMAT_ASTC_8x8_UNORM_BLOCK;
149  else if (header.blockdimX == 10 && header.blockdimY == 5) // 10x5
150  *pFormat = VK_FORMAT_ASTC_10x5_UNORM_BLOCK;
151  else if (header.blockdimX == 10 && header.blockdimY == 6) // 10x6
152  *pFormat = VK_FORMAT_ASTC_10x6_UNORM_BLOCK;
153  else if (header.blockdimX == 10 && header.blockdimY == 8) // 10x8
154  *pFormat = VK_FORMAT_ASTC_10x8_UNORM_BLOCK;
155  else if (header.blockdimX == 10 && header.blockdimY == 10) // 10x10
156  *pFormat = VK_FORMAT_ASTC_10x10_UNORM_BLOCK;
157  else if (header.blockdimX == 12 && header.blockdimY == 10) // 12x10
158  *pFormat = VK_FORMAT_ASTC_12x10_UNORM_BLOCK;
159  else if (header.blockdimX == 12 && header.blockdimY == 12) // 12x12
160  *pFormat = VK_FORMAT_ASTC_12x12_UNORM_BLOCK;
161  else
162  {
163  LOGE("Unknown ASTC block size %u x %u.\n", header.blockdimX, header.blockdimY);
164  return RESULT_ERROR_GENERIC;
165  }
166 
167  pBuffer->clear();
168  pBuffer->insert(end(*pBuffer), begin(compressed) + sizeof(ASTCHeader), end(compressed));
169  *pWidth = header.xsize[0] | (header.xsize[1] << 8) | (header.xsize[2] << 16);
170  *pHeight = header.ysize[0] | (header.ysize[1] << 8) | (header.ysize[2] << 16);
171  return RESULT_SUCCESS;
172 }
173 }
uint8_t magic[4]
Magic value.
Definition: assets.cpp:87
uint8_t blockdimY
Block size in Y.
Definition: assets.cpp:91
uint8_t xsize[3]
Size of the image in pixels (X), least significant byte first.
Definition: assets.cpp:95
uint8_t ysize[3]
Size of the image in pixels (Y), least significant byte first.
Definition: assets.cpp:97
uint8_t blockdimZ
Block size in Z.
Definition: assets.cpp:93
uint8_t blockdimX
Block size in X.
Definition: assets.cpp:89
Header for the on-disk format generated by astcenc.
Definition: assets.cpp:84