blob: 58f82f162531dd0452f42615637ade35a95c0d8f [file] [log] [blame]
Jon Ashburnbe582642014-12-22 12:04:40 -07001/**************************************************************************
2 *
Jon Ashburne922f712015-11-03 13:41:23 -07003 * Copyright 2014 Valve Software
Michael Lentine695f2c22015-09-09 12:39:13 -07004 * Copyright 2015 Google Inc.
Jon Ashburnbe582642014-12-22 12:04:40 -07005 * All Rights Reserved.
6 *
7 * Permission is hereby granted, free of charge, to any person obtaining a copy
8 * of this software and associated documentation files (the "Software"), to deal
9 * in the Software without restriction, including without limitation the rights
10 * to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
11 * copies of the Software, and to permit persons to whom the Software is
12 * furnished to do so, subject to the following conditions:
13 *
14 * The above copyright notice and this permission notice shall be included in
15 * all copies or substantial portions of the Software.
16 *
17 * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
18 * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
19 * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
20 * AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
21 * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
22 * OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
23 * THE SOFTWARE.
24 *
Jon Ashburne922f712015-11-03 13:41:23 -070025 * Author: Jon Ashburn <jon@lunarg.com>
26 * Author: Courtney Goeltzenleuchter <courtney@LunarG.com>
27 * Author: Tobin Ehlis <tobin@lunarg.com>
Jon Ashburnbe582642014-12-22 12:04:40 -070028 **************************************************************************/
29#include <fstream>
30#include <string>
31#include <map>
32#include <string.h>
David Pinedo9316d3b2015-11-06 12:54:48 -070033#include <vulkan/vk_layer.h>
Tobin Ehlisb1df55e2015-09-15 09:55:54 -060034#include <iostream>
Tobin Ehlisa0cb02e2015-07-03 10:15:26 -060035#include "vk_layer_config.h"
David Pinedo9316d3b2015-11-06 12:54:48 -070036#include "vulkan/vk_sdk_platform.h"
Jon Ashburnbe582642014-12-22 12:04:40 -070037
38#define MAX_CHARS_PER_LINE 4096
39
Mark Lobodzinskib39d9e62016-02-02 17:06:29 -070040class ConfigFile {
41 public:
Jon Ashburnbe582642014-12-22 12:04:40 -070042 ConfigFile();
43 ~ConfigFile();
44
45 const char *getOption(const std::string &_option);
Jon Ashburndec60512015-01-13 17:24:01 -070046 void setOption(const std::string &_option, const std::string &_val);
47
Mark Lobodzinskib39d9e62016-02-02 17:06:29 -070048 private:
Jon Ashburnbe582642014-12-22 12:04:40 -070049 bool m_fileIsParsed;
50 std::map<std::string, std::string> m_valueMap;
51
52 void parseFile(const char *filename);
53};
54
55static ConfigFile g_configFileObj;
Jon Ashburndec60512015-01-13 17:24:01 -070056
Mark Lobodzinskib39d9e62016-02-02 17:06:29 -070057static VkLayerDbgAction stringToDbgAction(const char *_enum) {
Jon Ashburndec60512015-01-13 17:24:01 -070058 // only handles single enum values
Courtney Goeltzenleuchterd8e229c2015-04-08 15:36:08 -060059 if (!strcmp(_enum, "VK_DBG_LAYER_ACTION_IGNORE"))
60 return VK_DBG_LAYER_ACTION_IGNORE;
Courtney Goeltzenleuchterd8e229c2015-04-08 15:36:08 -060061 else if (!strcmp(_enum, "VK_DBG_LAYER_ACTION_LOG_MSG"))
62 return VK_DBG_LAYER_ACTION_LOG_MSG;
Courtney Goeltzenleuchter5907ac42015-10-05 14:41:34 -060063#ifdef WIN32
64 else if (!strcmp(_enum, "VK_DBG_LAYER_ACTION_DEBUG_OUTPUT"))
65 return VK_DBG_LAYER_ACTION_DEBUG_OUTPUT;
66#endif
Courtney Goeltzenleuchterd8e229c2015-04-08 15:36:08 -060067 else if (!strcmp(_enum, "VK_DBG_LAYER_ACTION_BREAK"))
68 return VK_DBG_LAYER_ACTION_BREAK;
Mark Lobodzinskib39d9e62016-02-02 17:06:29 -070069 return (VkLayerDbgAction)0;
Jon Ashburndec60512015-01-13 17:24:01 -070070}
Courtney Goeltzenleuchterf579fa62015-06-10 17:39:03 -060071
Mark Lobodzinskib39d9e62016-02-02 17:06:29 -070072static VkFlags stringToDbgReportFlags(const char *_enum) {
Courtney Goeltzenleuchterf579fa62015-06-10 17:39:03 -060073 // only handles single enum values
Courtney Goeltzenleuchterfd4830c2015-11-25 10:30:56 -070074 if (!strcmp(_enum, "VK_DEBUG_REPORT_INFO"))
Courtney Goeltzenleuchter7415d5a2015-12-09 15:48:16 -070075 return VK_DEBUG_REPORT_INFO_BIT_EXT;
Courtney Goeltzenleuchterfd4830c2015-11-25 10:30:56 -070076 else if (!strcmp(_enum, "VK_DEBUG_REPORT_WARN"))
Courtney Goeltzenleuchter7415d5a2015-12-09 15:48:16 -070077 return VK_DEBUG_REPORT_WARN_BIT_EXT;
Courtney Goeltzenleuchterfd4830c2015-11-25 10:30:56 -070078 else if (!strcmp(_enum, "VK_DEBUG_REPORT_PERF_WARN"))
Courtney Goeltzenleuchter7415d5a2015-12-09 15:48:16 -070079 return VK_DEBUG_REPORT_PERF_WARN_BIT_EXT;
Courtney Goeltzenleuchterfd4830c2015-11-25 10:30:56 -070080 else if (!strcmp(_enum, "VK_DEBUG_REPORT_ERROR"))
Courtney Goeltzenleuchter7415d5a2015-12-09 15:48:16 -070081 return VK_DEBUG_REPORT_ERROR_BIT_EXT;
Courtney Goeltzenleuchterfd4830c2015-11-25 10:30:56 -070082 else if (!strcmp(_enum, "VK_DEBUG_REPORT_DEBUG"))
Courtney Goeltzenleuchter7415d5a2015-12-09 15:48:16 -070083 return VK_DEBUG_REPORT_DEBUG_BIT_EXT;
Mark Lobodzinskib39d9e62016-02-02 17:06:29 -070084 return (VkFlags)0;
Courtney Goeltzenleuchterf579fa62015-06-10 17:39:03 -060085}
86
Mark Lobodzinskib39d9e62016-02-02 17:06:29 -070087static unsigned int convertStringEnumVal(const char *_enum) {
Courtney Goeltzenleuchterf579fa62015-06-10 17:39:03 -060088 unsigned int ret;
89
90 ret = stringToDbgAction(_enum);
91 if (ret)
92 return ret;
93
94 return stringToDbgReportFlags(_enum);
95}
96
Mark Lobodzinskib39d9e62016-02-02 17:06:29 -070097const char *getLayerOption(const char *_option) {
Jon Ashburnbe582642014-12-22 12:04:40 -070098 return g_configFileObj.getOption(_option);
99}
100
Tobin Ehlisb1df55e2015-09-15 09:55:54 -0600101// If option is NULL or stdout, return stdout, otherwise try to open option
102// as a filename. If successful, return file handle, otherwise stdout
Mark Lobodzinskib39d9e62016-02-02 17:06:29 -0700103FILE *getLayerLogOutput(const char *_option, const char *layerName) {
104 FILE *log_output = NULL;
Tobin Ehlisb1df55e2015-09-15 09:55:54 -0600105 if (!_option || !strcmp("stdout", _option))
106 log_output = stdout;
107 else {
108 log_output = fopen(_option, "w");
109 if (log_output == NULL) {
110 if (_option)
Mark Lobodzinskib39d9e62016-02-02 17:06:29 -0700111 std::cout << std::endl << layerName
112 << " ERROR: Bad output filename specified: "
113 << _option << ". Writing to STDOUT instead"
114 << std::endl << std::endl;
Tobin Ehlisb1df55e2015-09-15 09:55:54 -0600115 log_output = stdout;
116 }
117 }
Cody Northrope3e39ef2015-09-16 08:35:29 -0600118 return log_output;
Tobin Ehlisb1df55e2015-09-15 09:55:54 -0600119}
120
Mark Lobodzinskib39d9e62016-02-02 17:06:29 -0700121VkDebugReportFlagsEXT getLayerOptionFlags(const char *_option,
122 uint32_t optionDefault) {
Courtney Goeltzenleuchter7415d5a2015-12-09 15:48:16 -0700123 VkDebugReportFlagsEXT flags = optionDefault;
Courtney Goeltzenleuchterb874b8c2015-06-14 11:34:49 -0600124 const char *option = (g_configFileObj.getOption(_option));
125
126 /* parse comma-separated options */
127 while (option) {
128 const char *p = strchr(option, ',');
129 size_t len;
130
131 if (p)
132 len = p - option;
133 else
134 len = strlen(option);
135
136 if (len > 0) {
137 if (strncmp(option, "warn", len) == 0) {
Courtney Goeltzenleuchter7415d5a2015-12-09 15:48:16 -0700138 flags |= VK_DEBUG_REPORT_WARN_BIT_EXT;
Courtney Goeltzenleuchterb874b8c2015-06-14 11:34:49 -0600139 } else if (strncmp(option, "info", len) == 0) {
Courtney Goeltzenleuchter7415d5a2015-12-09 15:48:16 -0700140 flags |= VK_DEBUG_REPORT_INFO_BIT_EXT;
Courtney Goeltzenleuchterb874b8c2015-06-14 11:34:49 -0600141 } else if (strncmp(option, "perf", len) == 0) {
Courtney Goeltzenleuchter7415d5a2015-12-09 15:48:16 -0700142 flags |= VK_DEBUG_REPORT_PERF_WARN_BIT_EXT;
Courtney Goeltzenleuchterb874b8c2015-06-14 11:34:49 -0600143 } else if (strncmp(option, "error", len) == 0) {
Courtney Goeltzenleuchter7415d5a2015-12-09 15:48:16 -0700144 flags |= VK_DEBUG_REPORT_ERROR_BIT_EXT;
Courtney Goeltzenleuchterb874b8c2015-06-14 11:34:49 -0600145 } else if (strncmp(option, "debug", len) == 0) {
Courtney Goeltzenleuchter7415d5a2015-12-09 15:48:16 -0700146 flags |= VK_DEBUG_REPORT_DEBUG_BIT_EXT;
Courtney Goeltzenleuchterb874b8c2015-06-14 11:34:49 -0600147 }
148 }
149
150 if (!p)
151 break;
152
153 option = p + 1;
154 }
155 return flags;
156}
157
Mark Lobodzinskib39d9e62016-02-02 17:06:29 -0700158bool getLayerOptionEnum(const char *_option, uint32_t *optionDefault) {
Jon Ashburna8aa8372015-03-03 15:07:15 -0700159 bool res;
Mark Lobodzinski8a2b75b2015-02-25 12:23:20 -0600160 const char *option = (g_configFileObj.getOption(_option));
161 if (option != NULL) {
Jon Ashburna8aa8372015-03-03 15:07:15 -0700162 *optionDefault = convertStringEnumVal(option);
163 res = false;
Mark Lobodzinski0d6e66d2015-02-25 15:14:06 -0600164 } else {
Jon Ashburna8aa8372015-03-03 15:07:15 -0700165 res = true;
Mark Lobodzinski8a2b75b2015-02-25 12:23:20 -0600166 }
Jon Ashburna8aa8372015-03-03 15:07:15 -0700167 return res;
Mark Lobodzinski8a2b75b2015-02-25 12:23:20 -0600168}
169
Mark Lobodzinskib39d9e62016-02-02 17:06:29 -0700170void setLayerOptionEnum(const char *_option, const char *_valEnum) {
Jon Ashburndec60512015-01-13 17:24:01 -0700171 unsigned int val = convertStringEnumVal(_valEnum);
172 char strVal[24];
173 snprintf(strVal, 24, "%u", val);
174 g_configFileObj.setOption(_option, strVal);
175}
176
Mark Lobodzinskib39d9e62016-02-02 17:06:29 -0700177void setLayerOption(const char *_option, const char *_val) {
Jon Ashburndec60512015-01-13 17:24:01 -0700178 g_configFileObj.setOption(_option, _val);
179}
180
Mark Lobodzinskib39d9e62016-02-02 17:06:29 -0700181ConfigFile::ConfigFile() : m_fileIsParsed(false) {}
Jon Ashburnbe582642014-12-22 12:04:40 -0700182
Mark Lobodzinskib39d9e62016-02-02 17:06:29 -0700183ConfigFile::~ConfigFile() {}
Jon Ashburnbe582642014-12-22 12:04:40 -0700184
Mark Lobodzinskib39d9e62016-02-02 17:06:29 -0700185const char *ConfigFile::getOption(const std::string &_option) {
Jon Ashburnbe582642014-12-22 12:04:40 -0700186 std::map<std::string, std::string>::const_iterator it;
Mark Lobodzinskib39d9e62016-02-02 17:06:29 -0700187 if (!m_fileIsParsed) {
Courtney Goeltzenleuchterd8e229c2015-04-08 15:36:08 -0600188 parseFile("vk_layer_settings.txt");
Jon Ashburnbe582642014-12-22 12:04:40 -0700189 }
190
191 if ((it = m_valueMap.find(_option)) == m_valueMap.end())
192 return NULL;
193 else
194 return it->second.c_str();
195}
196
Mark Lobodzinskib39d9e62016-02-02 17:06:29 -0700197void ConfigFile::setOption(const std::string &_option,
198 const std::string &_val) {
199 if (!m_fileIsParsed) {
Courtney Goeltzenleuchterd8e229c2015-04-08 15:36:08 -0600200 parseFile("vk_layer_settings.txt");
Jon Ashburndec60512015-01-13 17:24:01 -0700201 }
202
203 m_valueMap[_option] = _val;
204}
205
Mark Lobodzinskib39d9e62016-02-02 17:06:29 -0700206void ConfigFile::parseFile(const char *filename) {
Jon Ashburnbe582642014-12-22 12:04:40 -0700207 std::ifstream file;
208 char buf[MAX_CHARS_PER_LINE];
209
210 m_fileIsParsed = true;
211 m_valueMap.clear();
212
213 file.open(filename);
214 if (!file.good())
215 return;
216
217 // read tokens from the file and form option, value pairs
218 file.getline(buf, MAX_CHARS_PER_LINE);
Mark Lobodzinskib39d9e62016-02-02 17:06:29 -0700219 while (!file.eof()) {
Jon Ashburnbe582642014-12-22 12:04:40 -0700220 char option[512];
221 char value[512];
222
223 char *pComment;
224
Mark Lobodzinskib39d9e62016-02-02 17:06:29 -0700225 // discard any comments delimited by '#' in the line
Jon Ashburnbe582642014-12-22 12:04:40 -0700226 pComment = strchr(buf, '#');
227 if (pComment)
228 *pComment = '\0';
229
Mark Lobodzinskib39d9e62016-02-02 17:06:29 -0700230 if (sscanf(buf, " %511[^\n\t =] = %511[^\n \t]", option, value) == 2) {
Jon Ashburnbe582642014-12-22 12:04:40 -0700231 std::string optStr(option);
232 std::string valStr(value);
233 m_valueMap[optStr] = valStr;
234 }
235 file.getline(buf, MAX_CHARS_PER_LINE);
236 }
237}
238
Mark Lobodzinskib39d9e62016-02-02 17:06:29 -0700239void print_msg_flags(VkFlags msgFlags, char *msg_flags) {
Courtney Goeltzenleuchterf85e3612015-06-14 11:33:06 -0600240 bool separator = false;
241
242 msg_flags[0] = 0;
Courtney Goeltzenleuchter7415d5a2015-12-09 15:48:16 -0700243 if (msgFlags & VK_DEBUG_REPORT_DEBUG_BIT_EXT) {
Courtney Goeltzenleuchterf85e3612015-06-14 11:33:06 -0600244 strcat(msg_flags, "DEBUG");
245 separator = true;
246 }
Courtney Goeltzenleuchter7415d5a2015-12-09 15:48:16 -0700247 if (msgFlags & VK_DEBUG_REPORT_INFO_BIT_EXT) {
Mark Lobodzinskib39d9e62016-02-02 17:06:29 -0700248 if (separator)
249 strcat(msg_flags, ",");
Courtney Goeltzenleuchterf85e3612015-06-14 11:33:06 -0600250 strcat(msg_flags, "INFO");
251 separator = true;
252 }
Courtney Goeltzenleuchter7415d5a2015-12-09 15:48:16 -0700253 if (msgFlags & VK_DEBUG_REPORT_WARN_BIT_EXT) {
Mark Lobodzinskib39d9e62016-02-02 17:06:29 -0700254 if (separator)
255 strcat(msg_flags, ",");
Courtney Goeltzenleuchterf85e3612015-06-14 11:33:06 -0600256 strcat(msg_flags, "WARN");
257 separator = true;
258 }
Courtney Goeltzenleuchter7415d5a2015-12-09 15:48:16 -0700259 if (msgFlags & VK_DEBUG_REPORT_PERF_WARN_BIT_EXT) {
Mark Lobodzinskib39d9e62016-02-02 17:06:29 -0700260 if (separator)
261 strcat(msg_flags, ",");
Courtney Goeltzenleuchterf85e3612015-06-14 11:33:06 -0600262 strcat(msg_flags, "PERF");
263 separator = true;
264 }
Courtney Goeltzenleuchter7415d5a2015-12-09 15:48:16 -0700265 if (msgFlags & VK_DEBUG_REPORT_ERROR_BIT_EXT) {
Mark Lobodzinskib39d9e62016-02-02 17:06:29 -0700266 if (separator)
267 strcat(msg_flags, ",");
Courtney Goeltzenleuchterf85e3612015-06-14 11:33:06 -0600268 strcat(msg_flags, "ERROR");
269 }
270}