1*61046927SAndroid Build Coastguard Worker /*
2*61046927SAndroid Build Coastguard Worker * XML DRI client-side driver configuration
3*61046927SAndroid Build Coastguard Worker * Copyright (C) 2003 Felix Kuehling
4*61046927SAndroid Build Coastguard Worker *
5*61046927SAndroid Build Coastguard Worker * Permission is hereby granted, free of charge, to any person obtaining a
6*61046927SAndroid Build Coastguard Worker * copy of this software and associated documentation files (the "Software"),
7*61046927SAndroid Build Coastguard Worker * to deal in the Software without restriction, including without limitation
8*61046927SAndroid Build Coastguard Worker * the rights to use, copy, modify, merge, publish, distribute, sublicense,
9*61046927SAndroid Build Coastguard Worker * and/or sell copies of the Software, and to permit persons to whom the
10*61046927SAndroid Build Coastguard Worker * Software is furnished to do so, subject to the following conditions:
11*61046927SAndroid Build Coastguard Worker *
12*61046927SAndroid Build Coastguard Worker * The above copyright notice and this permission notice shall be included
13*61046927SAndroid Build Coastguard Worker * in all copies or substantial portions of the Software.
14*61046927SAndroid Build Coastguard Worker *
15*61046927SAndroid Build Coastguard Worker * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS
16*61046927SAndroid Build Coastguard Worker * OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
17*61046927SAndroid Build Coastguard Worker * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL
18*61046927SAndroid Build Coastguard Worker * FELIX KUEHLING, OR ANY OTHER CONTRIBUTORS BE LIABLE FOR ANY CLAIM,
19*61046927SAndroid Build Coastguard Worker * DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR
20*61046927SAndroid Build Coastguard Worker * OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE
21*61046927SAndroid Build Coastguard Worker * OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
22*61046927SAndroid Build Coastguard Worker *
23*61046927SAndroid Build Coastguard Worker */
24*61046927SAndroid Build Coastguard Worker /**
25*61046927SAndroid Build Coastguard Worker * \file xmlconfig.c
26*61046927SAndroid Build Coastguard Worker * \brief Driver-independent client-side part of the XML configuration
27*61046927SAndroid Build Coastguard Worker * \author Felix Kuehling
28*61046927SAndroid Build Coastguard Worker */
29*61046927SAndroid Build Coastguard Worker
30*61046927SAndroid Build Coastguard Worker #include "xmlconfig.h"
31*61046927SAndroid Build Coastguard Worker #include <limits.h>
32*61046927SAndroid Build Coastguard Worker #include <stdarg.h>
33*61046927SAndroid Build Coastguard Worker #include <stdbool.h>
34*61046927SAndroid Build Coastguard Worker #include <stdint.h>
35*61046927SAndroid Build Coastguard Worker #include <stdio.h>
36*61046927SAndroid Build Coastguard Worker #include <stdlib.h>
37*61046927SAndroid Build Coastguard Worker #include <string.h>
38*61046927SAndroid Build Coastguard Worker #include <assert.h>
39*61046927SAndroid Build Coastguard Worker #if WITH_XMLCONFIG
40*61046927SAndroid Build Coastguard Worker #include <expat.h>
41*61046927SAndroid Build Coastguard Worker #include <unistd.h>
42*61046927SAndroid Build Coastguard Worker #include <errno.h>
43*61046927SAndroid Build Coastguard Worker #include <dirent.h>
44*61046927SAndroid Build Coastguard Worker #include <sys/stat.h>
45*61046927SAndroid Build Coastguard Worker #endif
46*61046927SAndroid Build Coastguard Worker #ifdef NO_REGEX
47*61046927SAndroid Build Coastguard Worker typedef int regex_t;
48*61046927SAndroid Build Coastguard Worker #define REG_EXTENDED 0
49*61046927SAndroid Build Coastguard Worker #define REG_NOSUB 0
50*61046927SAndroid Build Coastguard Worker #define REG_NOMATCH 1
regcomp(regex_t * r,const char * s,int f)51*61046927SAndroid Build Coastguard Worker static inline int regcomp(regex_t *r, const char *s, int f) { return 0; }
regexec(regex_t * r,const char * s,int n,void * p,int f)52*61046927SAndroid Build Coastguard Worker static inline int regexec(regex_t *r, const char *s, int n, void *p, int f) { return REG_NOMATCH; }
regfree(regex_t * r)53*61046927SAndroid Build Coastguard Worker static inline void regfree(regex_t* r) {}
54*61046927SAndroid Build Coastguard Worker #else
55*61046927SAndroid Build Coastguard Worker #include <regex.h>
56*61046927SAndroid Build Coastguard Worker #endif
57*61046927SAndroid Build Coastguard Worker #include <fcntl.h>
58*61046927SAndroid Build Coastguard Worker #include <math.h>
59*61046927SAndroid Build Coastguard Worker #include "strndup.h"
60*61046927SAndroid Build Coastguard Worker #include "u_process.h"
61*61046927SAndroid Build Coastguard Worker #include "os_file.h"
62*61046927SAndroid Build Coastguard Worker #include "os_misc.h"
63*61046927SAndroid Build Coastguard Worker
64*61046927SAndroid Build Coastguard Worker /* For systems like Hurd */
65*61046927SAndroid Build Coastguard Worker #ifndef PATH_MAX
66*61046927SAndroid Build Coastguard Worker #define PATH_MAX 4096
67*61046927SAndroid Build Coastguard Worker #endif
68*61046927SAndroid Build Coastguard Worker
69*61046927SAndroid Build Coastguard Worker static bool
be_verbose(void)70*61046927SAndroid Build Coastguard Worker be_verbose(void)
71*61046927SAndroid Build Coastguard Worker {
72*61046927SAndroid Build Coastguard Worker const char *s = getenv("MESA_DEBUG");
73*61046927SAndroid Build Coastguard Worker if (!s)
74*61046927SAndroid Build Coastguard Worker return true;
75*61046927SAndroid Build Coastguard Worker
76*61046927SAndroid Build Coastguard Worker return strstr(s, "silent") == NULL;
77*61046927SAndroid Build Coastguard Worker }
78*61046927SAndroid Build Coastguard Worker
79*61046927SAndroid Build Coastguard Worker /** \brief Locale-independent integer parser.
80*61046927SAndroid Build Coastguard Worker *
81*61046927SAndroid Build Coastguard Worker * Works similar to strtol. Leading space is NOT skipped. The input
82*61046927SAndroid Build Coastguard Worker * number may have an optional sign. Radix is specified by base. If
83*61046927SAndroid Build Coastguard Worker * base is 0 then decimal is assumed unless the input number is
84*61046927SAndroid Build Coastguard Worker * prefixed by 0x or 0X for hexadecimal or 0 for octal. After
85*61046927SAndroid Build Coastguard Worker * returning tail points to the first character that is not part of
86*61046927SAndroid Build Coastguard Worker * the integer number. If no number was found then tail points to the
87*61046927SAndroid Build Coastguard Worker * start of the input string. */
88*61046927SAndroid Build Coastguard Worker static int
strToI(const char * string,const char ** tail,int base)89*61046927SAndroid Build Coastguard Worker strToI(const char *string, const char **tail, int base)
90*61046927SAndroid Build Coastguard Worker {
91*61046927SAndroid Build Coastguard Worker int radix = base == 0 ? 10 : base;
92*61046927SAndroid Build Coastguard Worker int result = 0;
93*61046927SAndroid Build Coastguard Worker int sign = 1;
94*61046927SAndroid Build Coastguard Worker bool numberFound = false;
95*61046927SAndroid Build Coastguard Worker const char *start = string;
96*61046927SAndroid Build Coastguard Worker
97*61046927SAndroid Build Coastguard Worker assert(radix >= 2 && radix <= 36);
98*61046927SAndroid Build Coastguard Worker
99*61046927SAndroid Build Coastguard Worker if (*string == '-') {
100*61046927SAndroid Build Coastguard Worker sign = -1;
101*61046927SAndroid Build Coastguard Worker string++;
102*61046927SAndroid Build Coastguard Worker } else if (*string == '+')
103*61046927SAndroid Build Coastguard Worker string++;
104*61046927SAndroid Build Coastguard Worker if (base == 0 && *string == '0') {
105*61046927SAndroid Build Coastguard Worker numberFound = true;
106*61046927SAndroid Build Coastguard Worker if (*(string+1) == 'x' || *(string+1) == 'X') {
107*61046927SAndroid Build Coastguard Worker radix = 16;
108*61046927SAndroid Build Coastguard Worker string += 2;
109*61046927SAndroid Build Coastguard Worker } else {
110*61046927SAndroid Build Coastguard Worker radix = 8;
111*61046927SAndroid Build Coastguard Worker string++;
112*61046927SAndroid Build Coastguard Worker }
113*61046927SAndroid Build Coastguard Worker }
114*61046927SAndroid Build Coastguard Worker do {
115*61046927SAndroid Build Coastguard Worker int digit = -1;
116*61046927SAndroid Build Coastguard Worker if (radix <= 10) {
117*61046927SAndroid Build Coastguard Worker if (*string >= '0' && *string < '0' + radix)
118*61046927SAndroid Build Coastguard Worker digit = *string - '0';
119*61046927SAndroid Build Coastguard Worker } else {
120*61046927SAndroid Build Coastguard Worker if (*string >= '0' && *string <= '9')
121*61046927SAndroid Build Coastguard Worker digit = *string - '0';
122*61046927SAndroid Build Coastguard Worker else if (*string >= 'a' && *string < 'a' + radix - 10)
123*61046927SAndroid Build Coastguard Worker digit = *string - 'a' + 10;
124*61046927SAndroid Build Coastguard Worker else if (*string >= 'A' && *string < 'A' + radix - 10)
125*61046927SAndroid Build Coastguard Worker digit = *string - 'A' + 10;
126*61046927SAndroid Build Coastguard Worker }
127*61046927SAndroid Build Coastguard Worker if (digit != -1) {
128*61046927SAndroid Build Coastguard Worker numberFound = true;
129*61046927SAndroid Build Coastguard Worker result = radix*result + digit;
130*61046927SAndroid Build Coastguard Worker string++;
131*61046927SAndroid Build Coastguard Worker } else
132*61046927SAndroid Build Coastguard Worker break;
133*61046927SAndroid Build Coastguard Worker } while (true);
134*61046927SAndroid Build Coastguard Worker *tail = numberFound ? string : start;
135*61046927SAndroid Build Coastguard Worker return sign * result;
136*61046927SAndroid Build Coastguard Worker }
137*61046927SAndroid Build Coastguard Worker
138*61046927SAndroid Build Coastguard Worker /** \brief Locale-independent floating-point parser.
139*61046927SAndroid Build Coastguard Worker *
140*61046927SAndroid Build Coastguard Worker * Works similar to strtod. Leading space is NOT skipped. The input
141*61046927SAndroid Build Coastguard Worker * number may have an optional sign. '.' is interpreted as decimal
142*61046927SAndroid Build Coastguard Worker * point and may occur at most once. Optionally the number may end in
143*61046927SAndroid Build Coastguard Worker * [eE]<exponent>, where <exponent> is an integer as recognized by
144*61046927SAndroid Build Coastguard Worker * strToI. In that case the result is number * 10^exponent. After
145*61046927SAndroid Build Coastguard Worker * returning tail points to the first character that is not part of
146*61046927SAndroid Build Coastguard Worker * the floating point number. If no number was found then tail points
147*61046927SAndroid Build Coastguard Worker * to the start of the input string.
148*61046927SAndroid Build Coastguard Worker *
149*61046927SAndroid Build Coastguard Worker * Uses two passes for maximum accuracy. */
150*61046927SAndroid Build Coastguard Worker static float
strToF(const char * string,const char ** tail)151*61046927SAndroid Build Coastguard Worker strToF(const char *string, const char **tail)
152*61046927SAndroid Build Coastguard Worker {
153*61046927SAndroid Build Coastguard Worker int nDigits = 0, pointPos, exponent;
154*61046927SAndroid Build Coastguard Worker float sign = 1.0f, result = 0.0f, scale;
155*61046927SAndroid Build Coastguard Worker const char *start = string, *numStart;
156*61046927SAndroid Build Coastguard Worker
157*61046927SAndroid Build Coastguard Worker /* sign */
158*61046927SAndroid Build Coastguard Worker if (*string == '-') {
159*61046927SAndroid Build Coastguard Worker sign = -1.0f;
160*61046927SAndroid Build Coastguard Worker string++;
161*61046927SAndroid Build Coastguard Worker } else if (*string == '+')
162*61046927SAndroid Build Coastguard Worker string++;
163*61046927SAndroid Build Coastguard Worker
164*61046927SAndroid Build Coastguard Worker /* first pass: determine position of decimal point, number of
165*61046927SAndroid Build Coastguard Worker * digits, exponent and the end of the number. */
166*61046927SAndroid Build Coastguard Worker numStart = string;
167*61046927SAndroid Build Coastguard Worker while (*string >= '0' && *string <= '9') {
168*61046927SAndroid Build Coastguard Worker string++;
169*61046927SAndroid Build Coastguard Worker nDigits++;
170*61046927SAndroid Build Coastguard Worker }
171*61046927SAndroid Build Coastguard Worker pointPos = nDigits;
172*61046927SAndroid Build Coastguard Worker if (*string == '.') {
173*61046927SAndroid Build Coastguard Worker string++;
174*61046927SAndroid Build Coastguard Worker while (*string >= '0' && *string <= '9') {
175*61046927SAndroid Build Coastguard Worker string++;
176*61046927SAndroid Build Coastguard Worker nDigits++;
177*61046927SAndroid Build Coastguard Worker }
178*61046927SAndroid Build Coastguard Worker }
179*61046927SAndroid Build Coastguard Worker if (nDigits == 0) {
180*61046927SAndroid Build Coastguard Worker /* no digits, no number */
181*61046927SAndroid Build Coastguard Worker *tail = start;
182*61046927SAndroid Build Coastguard Worker return 0.0f;
183*61046927SAndroid Build Coastguard Worker }
184*61046927SAndroid Build Coastguard Worker *tail = string;
185*61046927SAndroid Build Coastguard Worker if (*string == 'e' || *string == 'E') {
186*61046927SAndroid Build Coastguard Worker const char *expTail;
187*61046927SAndroid Build Coastguard Worker exponent = strToI(string+1, &expTail, 10);
188*61046927SAndroid Build Coastguard Worker if (expTail == string+1)
189*61046927SAndroid Build Coastguard Worker exponent = 0;
190*61046927SAndroid Build Coastguard Worker else
191*61046927SAndroid Build Coastguard Worker *tail = expTail;
192*61046927SAndroid Build Coastguard Worker } else
193*61046927SAndroid Build Coastguard Worker exponent = 0;
194*61046927SAndroid Build Coastguard Worker string = numStart;
195*61046927SAndroid Build Coastguard Worker
196*61046927SAndroid Build Coastguard Worker /* scale of the first digit */
197*61046927SAndroid Build Coastguard Worker scale = sign * (float)pow(10.0, (double)(pointPos-1 + exponent));
198*61046927SAndroid Build Coastguard Worker
199*61046927SAndroid Build Coastguard Worker /* second pass: parse digits */
200*61046927SAndroid Build Coastguard Worker do {
201*61046927SAndroid Build Coastguard Worker if (*string != '.') {
202*61046927SAndroid Build Coastguard Worker assert(*string >= '0' && *string <= '9');
203*61046927SAndroid Build Coastguard Worker result += scale * (float)(*string - '0');
204*61046927SAndroid Build Coastguard Worker scale *= 0.1f;
205*61046927SAndroid Build Coastguard Worker nDigits--;
206*61046927SAndroid Build Coastguard Worker }
207*61046927SAndroid Build Coastguard Worker string++;
208*61046927SAndroid Build Coastguard Worker } while (nDigits > 0);
209*61046927SAndroid Build Coastguard Worker
210*61046927SAndroid Build Coastguard Worker return result;
211*61046927SAndroid Build Coastguard Worker }
212*61046927SAndroid Build Coastguard Worker
213*61046927SAndroid Build Coastguard Worker /** \brief Parse a value of a given type. */
214*61046927SAndroid Build Coastguard Worker static unsigned char
parseValue(driOptionValue * v,driOptionType type,const char * string)215*61046927SAndroid Build Coastguard Worker parseValue(driOptionValue *v, driOptionType type, const char *string)
216*61046927SAndroid Build Coastguard Worker {
217*61046927SAndroid Build Coastguard Worker const char *tail = NULL;
218*61046927SAndroid Build Coastguard Worker /* skip leading white-space */
219*61046927SAndroid Build Coastguard Worker string += strspn(string, " \f\n\r\t\v");
220*61046927SAndroid Build Coastguard Worker switch (type) {
221*61046927SAndroid Build Coastguard Worker case DRI_BOOL:
222*61046927SAndroid Build Coastguard Worker if (!strcmp(string, "false")) {
223*61046927SAndroid Build Coastguard Worker v->_bool = false;
224*61046927SAndroid Build Coastguard Worker tail = string + 5;
225*61046927SAndroid Build Coastguard Worker } else if (!strcmp(string, "true")) {
226*61046927SAndroid Build Coastguard Worker v->_bool = true;
227*61046927SAndroid Build Coastguard Worker tail = string + 4;
228*61046927SAndroid Build Coastguard Worker }
229*61046927SAndroid Build Coastguard Worker else
230*61046927SAndroid Build Coastguard Worker return false;
231*61046927SAndroid Build Coastguard Worker break;
232*61046927SAndroid Build Coastguard Worker case DRI_ENUM: /* enum is just a special integer */
233*61046927SAndroid Build Coastguard Worker case DRI_INT:
234*61046927SAndroid Build Coastguard Worker v->_int = strToI(string, &tail, 0);
235*61046927SAndroid Build Coastguard Worker break;
236*61046927SAndroid Build Coastguard Worker case DRI_FLOAT:
237*61046927SAndroid Build Coastguard Worker v->_float = strToF(string, &tail);
238*61046927SAndroid Build Coastguard Worker break;
239*61046927SAndroid Build Coastguard Worker case DRI_STRING:
240*61046927SAndroid Build Coastguard Worker free(v->_string);
241*61046927SAndroid Build Coastguard Worker v->_string = strndup(string, STRING_CONF_MAXLEN);
242*61046927SAndroid Build Coastguard Worker return true;
243*61046927SAndroid Build Coastguard Worker case DRI_SECTION:
244*61046927SAndroid Build Coastguard Worker unreachable("shouldn't be parsing values in section declarations");
245*61046927SAndroid Build Coastguard Worker }
246*61046927SAndroid Build Coastguard Worker
247*61046927SAndroid Build Coastguard Worker if (tail == string)
248*61046927SAndroid Build Coastguard Worker return false; /* empty string (or containing only white-space) */
249*61046927SAndroid Build Coastguard Worker /* skip trailing white space */
250*61046927SAndroid Build Coastguard Worker if (*tail)
251*61046927SAndroid Build Coastguard Worker tail += strspn(tail, " \f\n\r\t\v");
252*61046927SAndroid Build Coastguard Worker if (*tail)
253*61046927SAndroid Build Coastguard Worker return false; /* something left over that is not part of value */
254*61046927SAndroid Build Coastguard Worker
255*61046927SAndroid Build Coastguard Worker return true;
256*61046927SAndroid Build Coastguard Worker }
257*61046927SAndroid Build Coastguard Worker
258*61046927SAndroid Build Coastguard Worker /** \brief Find an option in an option cache with the name as key */
259*61046927SAndroid Build Coastguard Worker static uint32_t
findOption(const driOptionCache * cache,const char * name)260*61046927SAndroid Build Coastguard Worker findOption(const driOptionCache *cache, const char *name)
261*61046927SAndroid Build Coastguard Worker {
262*61046927SAndroid Build Coastguard Worker uint32_t len = strlen(name);
263*61046927SAndroid Build Coastguard Worker uint32_t size = 1 << cache->tableSize, mask = size - 1;
264*61046927SAndroid Build Coastguard Worker uint32_t hash = 0;
265*61046927SAndroid Build Coastguard Worker uint32_t i, shift;
266*61046927SAndroid Build Coastguard Worker
267*61046927SAndroid Build Coastguard Worker /* compute a hash from the variable length name */
268*61046927SAndroid Build Coastguard Worker for (i = 0, shift = 0; i < len; ++i, shift = (shift+8) & 31)
269*61046927SAndroid Build Coastguard Worker hash += (uint32_t)name[i] << shift;
270*61046927SAndroid Build Coastguard Worker hash *= hash;
271*61046927SAndroid Build Coastguard Worker hash = (hash >> (16-cache->tableSize/2)) & mask;
272*61046927SAndroid Build Coastguard Worker
273*61046927SAndroid Build Coastguard Worker /* this is just the starting point of the linear search for the option */
274*61046927SAndroid Build Coastguard Worker for (i = 0; i < size; ++i, hash = (hash+1) & mask) {
275*61046927SAndroid Build Coastguard Worker /* if we hit an empty entry then the option is not defined (yet) */
276*61046927SAndroid Build Coastguard Worker if (cache->info[hash].name == NULL)
277*61046927SAndroid Build Coastguard Worker break;
278*61046927SAndroid Build Coastguard Worker else if (!strcmp(name, cache->info[hash].name))
279*61046927SAndroid Build Coastguard Worker break;
280*61046927SAndroid Build Coastguard Worker }
281*61046927SAndroid Build Coastguard Worker /* this assertion fails if the hash table is full */
282*61046927SAndroid Build Coastguard Worker assert (i < size);
283*61046927SAndroid Build Coastguard Worker
284*61046927SAndroid Build Coastguard Worker return hash;
285*61046927SAndroid Build Coastguard Worker }
286*61046927SAndroid Build Coastguard Worker
287*61046927SAndroid Build Coastguard Worker /** \brief Like strdup with error checking. */
288*61046927SAndroid Build Coastguard Worker #define XSTRDUP(dest,source) do { \
289*61046927SAndroid Build Coastguard Worker if (!(dest = strdup(source))) { \
290*61046927SAndroid Build Coastguard Worker fprintf(stderr, "%s: %d: out of memory.\n", __FILE__, __LINE__); \
291*61046927SAndroid Build Coastguard Worker abort(); \
292*61046927SAndroid Build Coastguard Worker } \
293*61046927SAndroid Build Coastguard Worker } while (0)
294*61046927SAndroid Build Coastguard Worker
295*61046927SAndroid Build Coastguard Worker /** \brief Check if a value is in info->range. */
296*61046927SAndroid Build Coastguard Worker UNUSED static bool
checkValue(const driOptionValue * v,const driOptionInfo * info)297*61046927SAndroid Build Coastguard Worker checkValue(const driOptionValue *v, const driOptionInfo *info)
298*61046927SAndroid Build Coastguard Worker {
299*61046927SAndroid Build Coastguard Worker switch (info->type) {
300*61046927SAndroid Build Coastguard Worker case DRI_ENUM: /* enum is just a special integer */
301*61046927SAndroid Build Coastguard Worker case DRI_INT:
302*61046927SAndroid Build Coastguard Worker return (info->range.start._int == info->range.end._int ||
303*61046927SAndroid Build Coastguard Worker (v->_int >= info->range.start._int &&
304*61046927SAndroid Build Coastguard Worker v->_int <= info->range.end._int));
305*61046927SAndroid Build Coastguard Worker
306*61046927SAndroid Build Coastguard Worker case DRI_FLOAT:
307*61046927SAndroid Build Coastguard Worker return (info->range.start._float == info->range.end._float ||
308*61046927SAndroid Build Coastguard Worker (v->_float >= info->range.start._float &&
309*61046927SAndroid Build Coastguard Worker v->_float <= info->range.end._float));
310*61046927SAndroid Build Coastguard Worker
311*61046927SAndroid Build Coastguard Worker default:
312*61046927SAndroid Build Coastguard Worker return true;
313*61046927SAndroid Build Coastguard Worker }
314*61046927SAndroid Build Coastguard Worker }
315*61046927SAndroid Build Coastguard Worker
316*61046927SAndroid Build Coastguard Worker void
driParseOptionInfo(driOptionCache * info,const driOptionDescription * configOptions,unsigned numOptions)317*61046927SAndroid Build Coastguard Worker driParseOptionInfo(driOptionCache *info,
318*61046927SAndroid Build Coastguard Worker const driOptionDescription *configOptions,
319*61046927SAndroid Build Coastguard Worker unsigned numOptions)
320*61046927SAndroid Build Coastguard Worker {
321*61046927SAndroid Build Coastguard Worker /* Make the hash table big enough to fit more than the maximum number of
322*61046927SAndroid Build Coastguard Worker * config options we've ever seen in a driver.
323*61046927SAndroid Build Coastguard Worker */
324*61046927SAndroid Build Coastguard Worker info->tableSize = 7;
325*61046927SAndroid Build Coastguard Worker info->info = calloc((size_t)1 << info->tableSize, sizeof(driOptionInfo));
326*61046927SAndroid Build Coastguard Worker info->values = calloc((size_t)1 << info->tableSize, sizeof(driOptionValue));
327*61046927SAndroid Build Coastguard Worker if (info->info == NULL || info->values == NULL) {
328*61046927SAndroid Build Coastguard Worker fprintf(stderr, "%s: %d: out of memory.\n", __FILE__, __LINE__);
329*61046927SAndroid Build Coastguard Worker abort();
330*61046927SAndroid Build Coastguard Worker }
331*61046927SAndroid Build Coastguard Worker
332*61046927SAndroid Build Coastguard Worker UNUSED bool in_section = false;
333*61046927SAndroid Build Coastguard Worker for (int o = 0; o < numOptions; o++) {
334*61046927SAndroid Build Coastguard Worker const driOptionDescription *opt = &configOptions[o];
335*61046927SAndroid Build Coastguard Worker
336*61046927SAndroid Build Coastguard Worker if (opt->info.type == DRI_SECTION) {
337*61046927SAndroid Build Coastguard Worker in_section = true;
338*61046927SAndroid Build Coastguard Worker continue;
339*61046927SAndroid Build Coastguard Worker }
340*61046927SAndroid Build Coastguard Worker
341*61046927SAndroid Build Coastguard Worker /* for driconf xml generation, options must always be preceded by a
342*61046927SAndroid Build Coastguard Worker * DRI_CONF_SECTION
343*61046927SAndroid Build Coastguard Worker */
344*61046927SAndroid Build Coastguard Worker assert(in_section);
345*61046927SAndroid Build Coastguard Worker
346*61046927SAndroid Build Coastguard Worker const char *name = opt->info.name;
347*61046927SAndroid Build Coastguard Worker int i = findOption(info, name);
348*61046927SAndroid Build Coastguard Worker driOptionInfo *optinfo = &info->info[i];
349*61046927SAndroid Build Coastguard Worker driOptionValue *optval = &info->values[i];
350*61046927SAndroid Build Coastguard Worker
351*61046927SAndroid Build Coastguard Worker if (optinfo->name) {
352*61046927SAndroid Build Coastguard Worker /* Duplicate options override the value, but the type must match. */
353*61046927SAndroid Build Coastguard Worker assert(optinfo->type == opt->info.type);
354*61046927SAndroid Build Coastguard Worker } else {
355*61046927SAndroid Build Coastguard Worker XSTRDUP(optinfo->name, name);
356*61046927SAndroid Build Coastguard Worker }
357*61046927SAndroid Build Coastguard Worker
358*61046927SAndroid Build Coastguard Worker optinfo->type = opt->info.type;
359*61046927SAndroid Build Coastguard Worker optinfo->range = opt->info.range;
360*61046927SAndroid Build Coastguard Worker
361*61046927SAndroid Build Coastguard Worker switch (opt->info.type) {
362*61046927SAndroid Build Coastguard Worker case DRI_BOOL:
363*61046927SAndroid Build Coastguard Worker optval->_bool = opt->value._bool;
364*61046927SAndroid Build Coastguard Worker break;
365*61046927SAndroid Build Coastguard Worker
366*61046927SAndroid Build Coastguard Worker case DRI_INT:
367*61046927SAndroid Build Coastguard Worker case DRI_ENUM:
368*61046927SAndroid Build Coastguard Worker optval->_int = opt->value._int;
369*61046927SAndroid Build Coastguard Worker break;
370*61046927SAndroid Build Coastguard Worker
371*61046927SAndroid Build Coastguard Worker case DRI_FLOAT:
372*61046927SAndroid Build Coastguard Worker optval->_float = opt->value._float;
373*61046927SAndroid Build Coastguard Worker break;
374*61046927SAndroid Build Coastguard Worker
375*61046927SAndroid Build Coastguard Worker case DRI_STRING:
376*61046927SAndroid Build Coastguard Worker XSTRDUP(optval->_string, opt->value._string);
377*61046927SAndroid Build Coastguard Worker break;
378*61046927SAndroid Build Coastguard Worker
379*61046927SAndroid Build Coastguard Worker case DRI_SECTION:
380*61046927SAndroid Build Coastguard Worker unreachable("handled above");
381*61046927SAndroid Build Coastguard Worker }
382*61046927SAndroid Build Coastguard Worker
383*61046927SAndroid Build Coastguard Worker /* Built-in default values should always be valid. */
384*61046927SAndroid Build Coastguard Worker assert(checkValue(optval, optinfo));
385*61046927SAndroid Build Coastguard Worker
386*61046927SAndroid Build Coastguard Worker const char *envVal = os_get_option(name);
387*61046927SAndroid Build Coastguard Worker if (envVal != NULL) {
388*61046927SAndroid Build Coastguard Worker driOptionValue v;
389*61046927SAndroid Build Coastguard Worker
390*61046927SAndroid Build Coastguard Worker /* make sure the value is initialized to something sensible */
391*61046927SAndroid Build Coastguard Worker v._string = NULL;
392*61046927SAndroid Build Coastguard Worker
393*61046927SAndroid Build Coastguard Worker if (parseValue(&v, opt->info.type, envVal) &&
394*61046927SAndroid Build Coastguard Worker checkValue(&v, optinfo)) {
395*61046927SAndroid Build Coastguard Worker /* don't use XML_WARNING, we want the user to see this! */
396*61046927SAndroid Build Coastguard Worker if (be_verbose()) {
397*61046927SAndroid Build Coastguard Worker fprintf(stderr,
398*61046927SAndroid Build Coastguard Worker "ATTENTION: default value of option %s overridden by environment.\n",
399*61046927SAndroid Build Coastguard Worker name);
400*61046927SAndroid Build Coastguard Worker }
401*61046927SAndroid Build Coastguard Worker *optval = v;
402*61046927SAndroid Build Coastguard Worker } else {
403*61046927SAndroid Build Coastguard Worker fprintf(stderr, "illegal environment value for %s: \"%s\". Ignoring.\n",
404*61046927SAndroid Build Coastguard Worker name, envVal);
405*61046927SAndroid Build Coastguard Worker }
406*61046927SAndroid Build Coastguard Worker }
407*61046927SAndroid Build Coastguard Worker }
408*61046927SAndroid Build Coastguard Worker }
409*61046927SAndroid Build Coastguard Worker
410*61046927SAndroid Build Coastguard Worker char *
driGetOptionsXml(const driOptionDescription * configOptions,unsigned numOptions)411*61046927SAndroid Build Coastguard Worker driGetOptionsXml(const driOptionDescription *configOptions, unsigned numOptions)
412*61046927SAndroid Build Coastguard Worker {
413*61046927SAndroid Build Coastguard Worker char *str = ralloc_strdup(NULL,
414*61046927SAndroid Build Coastguard Worker "<?xml version=\"1.0\" standalone=\"yes\"?>\n" \
415*61046927SAndroid Build Coastguard Worker "<!DOCTYPE driinfo [\n" \
416*61046927SAndroid Build Coastguard Worker " <!ELEMENT driinfo (section*)>\n" \
417*61046927SAndroid Build Coastguard Worker " <!ELEMENT section (description+, option+)>\n" \
418*61046927SAndroid Build Coastguard Worker " <!ELEMENT description (enum*)>\n" \
419*61046927SAndroid Build Coastguard Worker " <!ATTLIST description lang CDATA #FIXED \"en\"\n" \
420*61046927SAndroid Build Coastguard Worker " text CDATA #REQUIRED>\n" \
421*61046927SAndroid Build Coastguard Worker " <!ELEMENT option (description+)>\n" \
422*61046927SAndroid Build Coastguard Worker " <!ATTLIST option name CDATA #REQUIRED\n" \
423*61046927SAndroid Build Coastguard Worker " type (bool|enum|int|float) #REQUIRED\n" \
424*61046927SAndroid Build Coastguard Worker " default CDATA #REQUIRED\n" \
425*61046927SAndroid Build Coastguard Worker " valid CDATA #IMPLIED>\n" \
426*61046927SAndroid Build Coastguard Worker " <!ELEMENT enum EMPTY>\n" \
427*61046927SAndroid Build Coastguard Worker " <!ATTLIST enum value CDATA #REQUIRED\n" \
428*61046927SAndroid Build Coastguard Worker " text CDATA #REQUIRED>\n" \
429*61046927SAndroid Build Coastguard Worker "]>" \
430*61046927SAndroid Build Coastguard Worker "<driinfo>\n");
431*61046927SAndroid Build Coastguard Worker
432*61046927SAndroid Build Coastguard Worker bool in_section = false;
433*61046927SAndroid Build Coastguard Worker for (int o = 0; o < numOptions; o++) {
434*61046927SAndroid Build Coastguard Worker const driOptionDescription *opt = &configOptions[o];
435*61046927SAndroid Build Coastguard Worker
436*61046927SAndroid Build Coastguard Worker const char *name = opt->info.name;
437*61046927SAndroid Build Coastguard Worker const char *types[] = {
438*61046927SAndroid Build Coastguard Worker [DRI_BOOL] = "bool",
439*61046927SAndroid Build Coastguard Worker [DRI_INT] = "int",
440*61046927SAndroid Build Coastguard Worker [DRI_FLOAT] = "float",
441*61046927SAndroid Build Coastguard Worker [DRI_ENUM] = "enum",
442*61046927SAndroid Build Coastguard Worker [DRI_STRING] = "string",
443*61046927SAndroid Build Coastguard Worker };
444*61046927SAndroid Build Coastguard Worker
445*61046927SAndroid Build Coastguard Worker if (opt->info.type == DRI_SECTION) {
446*61046927SAndroid Build Coastguard Worker if (in_section)
447*61046927SAndroid Build Coastguard Worker ralloc_asprintf_append(&str, " </section>\n");
448*61046927SAndroid Build Coastguard Worker
449*61046927SAndroid Build Coastguard Worker ralloc_asprintf_append(&str,
450*61046927SAndroid Build Coastguard Worker " <section>\n"
451*61046927SAndroid Build Coastguard Worker " <description lang=\"en\" text=\"%s\"/>\n",
452*61046927SAndroid Build Coastguard Worker opt->desc);
453*61046927SAndroid Build Coastguard Worker
454*61046927SAndroid Build Coastguard Worker in_section = true;
455*61046927SAndroid Build Coastguard Worker continue;
456*61046927SAndroid Build Coastguard Worker }
457*61046927SAndroid Build Coastguard Worker
458*61046927SAndroid Build Coastguard Worker ralloc_asprintf_append(&str,
459*61046927SAndroid Build Coastguard Worker " <option name=\"%s\" type=\"%s\" default=\"",
460*61046927SAndroid Build Coastguard Worker name,
461*61046927SAndroid Build Coastguard Worker types[opt->info.type]);
462*61046927SAndroid Build Coastguard Worker
463*61046927SAndroid Build Coastguard Worker switch (opt->info.type) {
464*61046927SAndroid Build Coastguard Worker case DRI_BOOL:
465*61046927SAndroid Build Coastguard Worker ralloc_asprintf_append(&str, opt->value._bool ? "true" : "false");
466*61046927SAndroid Build Coastguard Worker break;
467*61046927SAndroid Build Coastguard Worker
468*61046927SAndroid Build Coastguard Worker case DRI_INT:
469*61046927SAndroid Build Coastguard Worker case DRI_ENUM:
470*61046927SAndroid Build Coastguard Worker ralloc_asprintf_append(&str, "%d", opt->value._int);
471*61046927SAndroid Build Coastguard Worker break;
472*61046927SAndroid Build Coastguard Worker
473*61046927SAndroid Build Coastguard Worker case DRI_FLOAT:
474*61046927SAndroid Build Coastguard Worker ralloc_asprintf_append(&str, "%f", opt->value._float);
475*61046927SAndroid Build Coastguard Worker break;
476*61046927SAndroid Build Coastguard Worker
477*61046927SAndroid Build Coastguard Worker case DRI_STRING:
478*61046927SAndroid Build Coastguard Worker ralloc_asprintf_append(&str, "%s", opt->value._string);
479*61046927SAndroid Build Coastguard Worker break;
480*61046927SAndroid Build Coastguard Worker
481*61046927SAndroid Build Coastguard Worker case DRI_SECTION:
482*61046927SAndroid Build Coastguard Worker unreachable("handled above");
483*61046927SAndroid Build Coastguard Worker break;
484*61046927SAndroid Build Coastguard Worker }
485*61046927SAndroid Build Coastguard Worker ralloc_asprintf_append(&str, "\"");
486*61046927SAndroid Build Coastguard Worker
487*61046927SAndroid Build Coastguard Worker
488*61046927SAndroid Build Coastguard Worker switch (opt->info.type) {
489*61046927SAndroid Build Coastguard Worker case DRI_INT:
490*61046927SAndroid Build Coastguard Worker case DRI_ENUM:
491*61046927SAndroid Build Coastguard Worker if (opt->info.range.start._int < opt->info.range.end._int) {
492*61046927SAndroid Build Coastguard Worker ralloc_asprintf_append(&str, " valid=\"%d:%d\"",
493*61046927SAndroid Build Coastguard Worker opt->info.range.start._int,
494*61046927SAndroid Build Coastguard Worker opt->info.range.end._int);
495*61046927SAndroid Build Coastguard Worker }
496*61046927SAndroid Build Coastguard Worker break;
497*61046927SAndroid Build Coastguard Worker
498*61046927SAndroid Build Coastguard Worker case DRI_FLOAT:
499*61046927SAndroid Build Coastguard Worker if (opt->info.range.start._float < opt->info.range.end._float) {
500*61046927SAndroid Build Coastguard Worker ralloc_asprintf_append(&str, " valid=\"%f:%f\"",
501*61046927SAndroid Build Coastguard Worker opt->info.range.start._float,
502*61046927SAndroid Build Coastguard Worker opt->info.range.end._float);
503*61046927SAndroid Build Coastguard Worker }
504*61046927SAndroid Build Coastguard Worker break;
505*61046927SAndroid Build Coastguard Worker
506*61046927SAndroid Build Coastguard Worker default:
507*61046927SAndroid Build Coastguard Worker break;
508*61046927SAndroid Build Coastguard Worker }
509*61046927SAndroid Build Coastguard Worker
510*61046927SAndroid Build Coastguard Worker ralloc_asprintf_append(&str, ">\n"); /* end of <option> */
511*61046927SAndroid Build Coastguard Worker
512*61046927SAndroid Build Coastguard Worker
513*61046927SAndroid Build Coastguard Worker ralloc_asprintf_append(&str, " <description lang=\"en\" text=\"%s\"%s>\n",
514*61046927SAndroid Build Coastguard Worker opt->desc, opt->info.type != DRI_ENUM ? "/" : "");
515*61046927SAndroid Build Coastguard Worker
516*61046927SAndroid Build Coastguard Worker if (opt->info.type == DRI_ENUM) {
517*61046927SAndroid Build Coastguard Worker for (int i = 0; i < ARRAY_SIZE(opt->enums) && opt->enums[i].desc; i++) {
518*61046927SAndroid Build Coastguard Worker ralloc_asprintf_append(&str, " <enum value=\"%d\" text=\"%s\"/>\n",
519*61046927SAndroid Build Coastguard Worker opt->enums[i].value, opt->enums[i].desc);
520*61046927SAndroid Build Coastguard Worker }
521*61046927SAndroid Build Coastguard Worker ralloc_asprintf_append(&str, " </description>\n");
522*61046927SAndroid Build Coastguard Worker }
523*61046927SAndroid Build Coastguard Worker
524*61046927SAndroid Build Coastguard Worker ralloc_asprintf_append(&str, " </option>\n");
525*61046927SAndroid Build Coastguard Worker }
526*61046927SAndroid Build Coastguard Worker
527*61046927SAndroid Build Coastguard Worker assert(in_section);
528*61046927SAndroid Build Coastguard Worker ralloc_asprintf_append(&str, " </section>\n");
529*61046927SAndroid Build Coastguard Worker
530*61046927SAndroid Build Coastguard Worker ralloc_asprintf_append(&str, "</driinfo>\n");
531*61046927SAndroid Build Coastguard Worker
532*61046927SAndroid Build Coastguard Worker char *output = strdup(str);
533*61046927SAndroid Build Coastguard Worker ralloc_free(str);
534*61046927SAndroid Build Coastguard Worker
535*61046927SAndroid Build Coastguard Worker return output;
536*61046927SAndroid Build Coastguard Worker }
537*61046927SAndroid Build Coastguard Worker
538*61046927SAndroid Build Coastguard Worker /**
539*61046927SAndroid Build Coastguard Worker * Print message to \c stderr if the \c LIBGL_DEBUG environment variable
540*61046927SAndroid Build Coastguard Worker * is set.
541*61046927SAndroid Build Coastguard Worker *
542*61046927SAndroid Build Coastguard Worker * Is called from the drivers.
543*61046927SAndroid Build Coastguard Worker *
544*61046927SAndroid Build Coastguard Worker * \param f \c printf like format string.
545*61046927SAndroid Build Coastguard Worker */
546*61046927SAndroid Build Coastguard Worker static void
__driUtilMessage(const char * f,...)547*61046927SAndroid Build Coastguard Worker __driUtilMessage(const char *f, ...)
548*61046927SAndroid Build Coastguard Worker {
549*61046927SAndroid Build Coastguard Worker va_list args;
550*61046927SAndroid Build Coastguard Worker const char *libgl_debug;
551*61046927SAndroid Build Coastguard Worker
552*61046927SAndroid Build Coastguard Worker libgl_debug=getenv("LIBGL_DEBUG");
553*61046927SAndroid Build Coastguard Worker if (libgl_debug && !strstr(libgl_debug, "quiet")) {
554*61046927SAndroid Build Coastguard Worker fprintf(stderr, "libGL: ");
555*61046927SAndroid Build Coastguard Worker va_start(args, f);
556*61046927SAndroid Build Coastguard Worker vfprintf(stderr, f, args);
557*61046927SAndroid Build Coastguard Worker va_end(args);
558*61046927SAndroid Build Coastguard Worker fprintf(stderr, "\n");
559*61046927SAndroid Build Coastguard Worker }
560*61046927SAndroid Build Coastguard Worker }
561*61046927SAndroid Build Coastguard Worker
562*61046927SAndroid Build Coastguard Worker /* We don't have real line/column # info in static-config case: */
563*61046927SAndroid Build Coastguard Worker #if !WITH_XML_CONFIG
564*61046927SAndroid Build Coastguard Worker # define XML_GetCurrentLineNumber(p) -1
565*61046927SAndroid Build Coastguard Worker # define XML_GetCurrentColumnNumber(p) -1
566*61046927SAndroid Build Coastguard Worker #endif
567*61046927SAndroid Build Coastguard Worker
568*61046927SAndroid Build Coastguard Worker /** \brief Output a warning message. */
569*61046927SAndroid Build Coastguard Worker #define XML_WARNING1(msg) do { \
570*61046927SAndroid Build Coastguard Worker __driUtilMessage("Warning in %s line %d, column %d: "msg, data->name, \
571*61046927SAndroid Build Coastguard Worker (int) XML_GetCurrentLineNumber(data->parser), \
572*61046927SAndroid Build Coastguard Worker (int) XML_GetCurrentColumnNumber(data->parser)); \
573*61046927SAndroid Build Coastguard Worker } while (0)
574*61046927SAndroid Build Coastguard Worker #define XML_WARNING(msg, ...) do { \
575*61046927SAndroid Build Coastguard Worker __driUtilMessage("Warning in %s line %d, column %d: "msg, data->name, \
576*61046927SAndroid Build Coastguard Worker (int) XML_GetCurrentLineNumber(data->parser), \
577*61046927SAndroid Build Coastguard Worker (int) XML_GetCurrentColumnNumber(data->parser), \
578*61046927SAndroid Build Coastguard Worker ##__VA_ARGS__); \
579*61046927SAndroid Build Coastguard Worker } while (0)
580*61046927SAndroid Build Coastguard Worker /** \brief Output an error message. */
581*61046927SAndroid Build Coastguard Worker #define XML_ERROR1(msg) do { \
582*61046927SAndroid Build Coastguard Worker __driUtilMessage("Error in %s line %d, column %d: "msg, data->name, \
583*61046927SAndroid Build Coastguard Worker (int) XML_GetCurrentLineNumber(data->parser), \
584*61046927SAndroid Build Coastguard Worker (int) XML_GetCurrentColumnNumber(data->parser)); \
585*61046927SAndroid Build Coastguard Worker } while (0)
586*61046927SAndroid Build Coastguard Worker #define XML_ERROR(msg, ...) do { \
587*61046927SAndroid Build Coastguard Worker __driUtilMessage("Error in %s line %d, column %d: "msg, data->name, \
588*61046927SAndroid Build Coastguard Worker (int) XML_GetCurrentLineNumber(data->parser), \
589*61046927SAndroid Build Coastguard Worker (int) XML_GetCurrentColumnNumber(data->parser), \
590*61046927SAndroid Build Coastguard Worker ##__VA_ARGS__); \
591*61046927SAndroid Build Coastguard Worker } while (0)
592*61046927SAndroid Build Coastguard Worker
593*61046927SAndroid Build Coastguard Worker /** \brief Parser context for configuration files. */
594*61046927SAndroid Build Coastguard Worker struct OptConfData {
595*61046927SAndroid Build Coastguard Worker const char *name;
596*61046927SAndroid Build Coastguard Worker #if WITH_XMLCONFIG
597*61046927SAndroid Build Coastguard Worker XML_Parser parser;
598*61046927SAndroid Build Coastguard Worker #endif
599*61046927SAndroid Build Coastguard Worker driOptionCache *cache;
600*61046927SAndroid Build Coastguard Worker int screenNum;
601*61046927SAndroid Build Coastguard Worker const char *driverName, *execName;
602*61046927SAndroid Build Coastguard Worker const char *kernelDriverName;
603*61046927SAndroid Build Coastguard Worker const char *deviceName;
604*61046927SAndroid Build Coastguard Worker const char *engineName;
605*61046927SAndroid Build Coastguard Worker const char *applicationName;
606*61046927SAndroid Build Coastguard Worker uint32_t engineVersion;
607*61046927SAndroid Build Coastguard Worker uint32_t applicationVersion;
608*61046927SAndroid Build Coastguard Worker uint32_t ignoringDevice;
609*61046927SAndroid Build Coastguard Worker uint32_t ignoringApp;
610*61046927SAndroid Build Coastguard Worker uint32_t inDriConf;
611*61046927SAndroid Build Coastguard Worker uint32_t inDevice;
612*61046927SAndroid Build Coastguard Worker uint32_t inApp;
613*61046927SAndroid Build Coastguard Worker uint32_t inOption;
614*61046927SAndroid Build Coastguard Worker };
615*61046927SAndroid Build Coastguard Worker
616*61046927SAndroid Build Coastguard Worker /** \brief Parse a list of ranges of type info->type. */
617*61046927SAndroid Build Coastguard Worker static unsigned char
parseRange(driOptionInfo * info,const char * string)618*61046927SAndroid Build Coastguard Worker parseRange(driOptionInfo *info, const char *string)
619*61046927SAndroid Build Coastguard Worker {
620*61046927SAndroid Build Coastguard Worker char *cp;
621*61046927SAndroid Build Coastguard Worker
622*61046927SAndroid Build Coastguard Worker XSTRDUP(cp, string);
623*61046927SAndroid Build Coastguard Worker
624*61046927SAndroid Build Coastguard Worker char *sep;
625*61046927SAndroid Build Coastguard Worker sep = strchr(cp, ':');
626*61046927SAndroid Build Coastguard Worker if (!sep) {
627*61046927SAndroid Build Coastguard Worker free(cp);
628*61046927SAndroid Build Coastguard Worker return false;
629*61046927SAndroid Build Coastguard Worker }
630*61046927SAndroid Build Coastguard Worker
631*61046927SAndroid Build Coastguard Worker *sep = '\0';
632*61046927SAndroid Build Coastguard Worker if (!parseValue(&info->range.start, info->type, cp) ||
633*61046927SAndroid Build Coastguard Worker !parseValue(&info->range.end, info->type, sep+1)) {
634*61046927SAndroid Build Coastguard Worker free(cp);
635*61046927SAndroid Build Coastguard Worker return false;
636*61046927SAndroid Build Coastguard Worker }
637*61046927SAndroid Build Coastguard Worker if (info->type == DRI_INT &&
638*61046927SAndroid Build Coastguard Worker info->range.start._int >= info->range.end._int) {
639*61046927SAndroid Build Coastguard Worker free(cp);
640*61046927SAndroid Build Coastguard Worker return false;
641*61046927SAndroid Build Coastguard Worker }
642*61046927SAndroid Build Coastguard Worker if (info->type == DRI_FLOAT &&
643*61046927SAndroid Build Coastguard Worker info->range.start._float >= info->range.end._float) {
644*61046927SAndroid Build Coastguard Worker free(cp);
645*61046927SAndroid Build Coastguard Worker return false;
646*61046927SAndroid Build Coastguard Worker }
647*61046927SAndroid Build Coastguard Worker
648*61046927SAndroid Build Coastguard Worker free(cp);
649*61046927SAndroid Build Coastguard Worker return true;
650*61046927SAndroid Build Coastguard Worker }
651*61046927SAndroid Build Coastguard Worker
652*61046927SAndroid Build Coastguard Worker /** \brief Parse attributes of a device element. */
653*61046927SAndroid Build Coastguard Worker static void
parseDeviceAttr(struct OptConfData * data,const char ** attr)654*61046927SAndroid Build Coastguard Worker parseDeviceAttr(struct OptConfData *data, const char **attr)
655*61046927SAndroid Build Coastguard Worker {
656*61046927SAndroid Build Coastguard Worker uint32_t i;
657*61046927SAndroid Build Coastguard Worker const char *driver = NULL, *screen = NULL, *kernel = NULL, *device = NULL;
658*61046927SAndroid Build Coastguard Worker for (i = 0; attr[i]; i += 2) {
659*61046927SAndroid Build Coastguard Worker if (!strcmp(attr[i], "driver")) driver = attr[i+1];
660*61046927SAndroid Build Coastguard Worker else if (!strcmp(attr[i], "screen")) screen = attr[i+1];
661*61046927SAndroid Build Coastguard Worker else if (!strcmp(attr[i], "kernel_driver")) kernel = attr[i+1];
662*61046927SAndroid Build Coastguard Worker else if (!strcmp(attr[i], "device")) device = attr[i+1];
663*61046927SAndroid Build Coastguard Worker else XML_WARNING("unknown device attribute: %s.", attr[i]);
664*61046927SAndroid Build Coastguard Worker }
665*61046927SAndroid Build Coastguard Worker if (driver && strcmp(driver, data->driverName))
666*61046927SAndroid Build Coastguard Worker data->ignoringDevice = data->inDevice;
667*61046927SAndroid Build Coastguard Worker else if (kernel && (!data->kernelDriverName ||
668*61046927SAndroid Build Coastguard Worker strcmp(kernel, data->kernelDriverName)))
669*61046927SAndroid Build Coastguard Worker data->ignoringDevice = data->inDevice;
670*61046927SAndroid Build Coastguard Worker else if (device && (!data->deviceName ||
671*61046927SAndroid Build Coastguard Worker strcmp(device, data->deviceName)))
672*61046927SAndroid Build Coastguard Worker data->ignoringDevice = data->inDevice;
673*61046927SAndroid Build Coastguard Worker else if (screen) {
674*61046927SAndroid Build Coastguard Worker driOptionValue screenNum;
675*61046927SAndroid Build Coastguard Worker if (!parseValue(&screenNum, DRI_INT, screen))
676*61046927SAndroid Build Coastguard Worker XML_WARNING("illegal screen number: %s.", screen);
677*61046927SAndroid Build Coastguard Worker else if (screenNum._int != data->screenNum)
678*61046927SAndroid Build Coastguard Worker data->ignoringDevice = data->inDevice;
679*61046927SAndroid Build Coastguard Worker }
680*61046927SAndroid Build Coastguard Worker }
681*61046927SAndroid Build Coastguard Worker
682*61046927SAndroid Build Coastguard Worker /** \brief Parse attributes of an application element. */
683*61046927SAndroid Build Coastguard Worker static void
parseAppAttr(struct OptConfData * data,const char ** attr)684*61046927SAndroid Build Coastguard Worker parseAppAttr(struct OptConfData *data, const char **attr)
685*61046927SAndroid Build Coastguard Worker {
686*61046927SAndroid Build Coastguard Worker uint32_t i;
687*61046927SAndroid Build Coastguard Worker const char *exec = NULL;
688*61046927SAndroid Build Coastguard Worker const char *sha1 = NULL;
689*61046927SAndroid Build Coastguard Worker const char *exec_regexp = NULL;
690*61046927SAndroid Build Coastguard Worker const char *application_name_match = NULL;
691*61046927SAndroid Build Coastguard Worker const char *application_versions = NULL;
692*61046927SAndroid Build Coastguard Worker driOptionInfo version_range = {
693*61046927SAndroid Build Coastguard Worker .type = DRI_INT,
694*61046927SAndroid Build Coastguard Worker };
695*61046927SAndroid Build Coastguard Worker
696*61046927SAndroid Build Coastguard Worker for (i = 0; attr[i]; i += 2) {
697*61046927SAndroid Build Coastguard Worker if (!strcmp(attr[i], "name")) /* not needed here */;
698*61046927SAndroid Build Coastguard Worker else if (!strcmp(attr[i], "executable")) exec = attr[i+1];
699*61046927SAndroid Build Coastguard Worker else if (!strcmp(attr[i], "executable_regexp")) exec_regexp = attr[i+1];
700*61046927SAndroid Build Coastguard Worker else if (!strcmp(attr[i], "sha1")) sha1 = attr[i+1];
701*61046927SAndroid Build Coastguard Worker else if (!strcmp(attr[i], "application_name_match"))
702*61046927SAndroid Build Coastguard Worker application_name_match = attr[i+1];
703*61046927SAndroid Build Coastguard Worker else if (!strcmp(attr[i], "application_versions"))
704*61046927SAndroid Build Coastguard Worker application_versions = attr[i+1];
705*61046927SAndroid Build Coastguard Worker else XML_WARNING("unknown application attribute: %s.", attr[i]);
706*61046927SAndroid Build Coastguard Worker }
707*61046927SAndroid Build Coastguard Worker if (exec && strcmp(exec, data->execName)) {
708*61046927SAndroid Build Coastguard Worker data->ignoringApp = data->inApp;
709*61046927SAndroid Build Coastguard Worker } else if (exec_regexp) {
710*61046927SAndroid Build Coastguard Worker regex_t re;
711*61046927SAndroid Build Coastguard Worker
712*61046927SAndroid Build Coastguard Worker if (regcomp(&re, exec_regexp, REG_EXTENDED|REG_NOSUB) == 0) {
713*61046927SAndroid Build Coastguard Worker if (regexec(&re, data->execName, 0, NULL, 0) == REG_NOMATCH)
714*61046927SAndroid Build Coastguard Worker data->ignoringApp = data->inApp;
715*61046927SAndroid Build Coastguard Worker regfree(&re);
716*61046927SAndroid Build Coastguard Worker } else
717*61046927SAndroid Build Coastguard Worker XML_WARNING("Invalid executable_regexp=\"%s\".", exec_regexp);
718*61046927SAndroid Build Coastguard Worker } else if (sha1) {
719*61046927SAndroid Build Coastguard Worker /* SHA1_DIGEST_STRING_LENGTH includes terminating null byte */
720*61046927SAndroid Build Coastguard Worker if (strlen(sha1) != (SHA1_DIGEST_STRING_LENGTH - 1)) {
721*61046927SAndroid Build Coastguard Worker XML_WARNING("Incorrect sha1 application attribute");
722*61046927SAndroid Build Coastguard Worker data->ignoringApp = data->inApp;
723*61046927SAndroid Build Coastguard Worker } else {
724*61046927SAndroid Build Coastguard Worker size_t len;
725*61046927SAndroid Build Coastguard Worker char* content;
726*61046927SAndroid Build Coastguard Worker char path[PATH_MAX];
727*61046927SAndroid Build Coastguard Worker if (util_get_process_exec_path(path, ARRAY_SIZE(path)) > 0 &&
728*61046927SAndroid Build Coastguard Worker (content = os_read_file(path, &len))) {
729*61046927SAndroid Build Coastguard Worker uint8_t sha1x[SHA1_DIGEST_LENGTH];
730*61046927SAndroid Build Coastguard Worker char sha1s[SHA1_DIGEST_STRING_LENGTH];
731*61046927SAndroid Build Coastguard Worker _mesa_sha1_compute(content, len, sha1x);
732*61046927SAndroid Build Coastguard Worker _mesa_sha1_format((char*) sha1s, sha1x);
733*61046927SAndroid Build Coastguard Worker free(content);
734*61046927SAndroid Build Coastguard Worker
735*61046927SAndroid Build Coastguard Worker if (strcmp(sha1, sha1s)) {
736*61046927SAndroid Build Coastguard Worker data->ignoringApp = data->inApp;
737*61046927SAndroid Build Coastguard Worker }
738*61046927SAndroid Build Coastguard Worker } else {
739*61046927SAndroid Build Coastguard Worker data->ignoringApp = data->inApp;
740*61046927SAndroid Build Coastguard Worker }
741*61046927SAndroid Build Coastguard Worker }
742*61046927SAndroid Build Coastguard Worker } else if (application_name_match) {
743*61046927SAndroid Build Coastguard Worker regex_t re;
744*61046927SAndroid Build Coastguard Worker
745*61046927SAndroid Build Coastguard Worker if (regcomp(&re, application_name_match, REG_EXTENDED|REG_NOSUB) == 0) {
746*61046927SAndroid Build Coastguard Worker if (regexec(&re, data->applicationName, 0, NULL, 0) == REG_NOMATCH)
747*61046927SAndroid Build Coastguard Worker data->ignoringApp = data->inApp;
748*61046927SAndroid Build Coastguard Worker regfree(&re);
749*61046927SAndroid Build Coastguard Worker } else
750*61046927SAndroid Build Coastguard Worker XML_WARNING("Invalid application_name_match=\"%s\".", application_name_match);
751*61046927SAndroid Build Coastguard Worker }
752*61046927SAndroid Build Coastguard Worker if (application_versions) {
753*61046927SAndroid Build Coastguard Worker driOptionValue v = { ._int = data->applicationVersion };
754*61046927SAndroid Build Coastguard Worker if (parseRange(&version_range, application_versions)) {
755*61046927SAndroid Build Coastguard Worker if (!checkValue(&v, &version_range))
756*61046927SAndroid Build Coastguard Worker data->ignoringApp = data->inApp;
757*61046927SAndroid Build Coastguard Worker } else {
758*61046927SAndroid Build Coastguard Worker XML_WARNING("Failed to parse application_versions range=\"%s\".",
759*61046927SAndroid Build Coastguard Worker application_versions);
760*61046927SAndroid Build Coastguard Worker }
761*61046927SAndroid Build Coastguard Worker }
762*61046927SAndroid Build Coastguard Worker }
763*61046927SAndroid Build Coastguard Worker
764*61046927SAndroid Build Coastguard Worker /** \brief Parse attributes of an application element. */
765*61046927SAndroid Build Coastguard Worker static void
parseEngineAttr(struct OptConfData * data,const char ** attr)766*61046927SAndroid Build Coastguard Worker parseEngineAttr(struct OptConfData *data, const char **attr)
767*61046927SAndroid Build Coastguard Worker {
768*61046927SAndroid Build Coastguard Worker uint32_t i;
769*61046927SAndroid Build Coastguard Worker const char *engine_name_match = NULL, *engine_versions = NULL;
770*61046927SAndroid Build Coastguard Worker driOptionInfo version_range = {
771*61046927SAndroid Build Coastguard Worker .type = DRI_INT,
772*61046927SAndroid Build Coastguard Worker };
773*61046927SAndroid Build Coastguard Worker for (i = 0; attr[i]; i += 2) {
774*61046927SAndroid Build Coastguard Worker if (!strcmp(attr[i], "name")) /* not needed here */;
775*61046927SAndroid Build Coastguard Worker else if (!strcmp(attr[i], "engine_name_match")) engine_name_match = attr[i+1];
776*61046927SAndroid Build Coastguard Worker else if (!strcmp(attr[i], "engine_versions")) engine_versions = attr[i+1];
777*61046927SAndroid Build Coastguard Worker else XML_WARNING("unknown application attribute: %s.", attr[i]);
778*61046927SAndroid Build Coastguard Worker }
779*61046927SAndroid Build Coastguard Worker if (engine_name_match) {
780*61046927SAndroid Build Coastguard Worker regex_t re;
781*61046927SAndroid Build Coastguard Worker
782*61046927SAndroid Build Coastguard Worker if (regcomp(&re, engine_name_match, REG_EXTENDED|REG_NOSUB) == 0) {
783*61046927SAndroid Build Coastguard Worker if (regexec(&re, data->engineName, 0, NULL, 0) == REG_NOMATCH)
784*61046927SAndroid Build Coastguard Worker data->ignoringApp = data->inApp;
785*61046927SAndroid Build Coastguard Worker regfree(&re);
786*61046927SAndroid Build Coastguard Worker } else
787*61046927SAndroid Build Coastguard Worker XML_WARNING("Invalid engine_name_match=\"%s\".", engine_name_match);
788*61046927SAndroid Build Coastguard Worker }
789*61046927SAndroid Build Coastguard Worker if (engine_versions) {
790*61046927SAndroid Build Coastguard Worker driOptionValue v = { ._int = data->engineVersion };
791*61046927SAndroid Build Coastguard Worker if (parseRange(&version_range, engine_versions)) {
792*61046927SAndroid Build Coastguard Worker if (!checkValue(&v, &version_range))
793*61046927SAndroid Build Coastguard Worker data->ignoringApp = data->inApp;
794*61046927SAndroid Build Coastguard Worker } else {
795*61046927SAndroid Build Coastguard Worker XML_WARNING("Failed to parse engine_versions range=\"%s\".",
796*61046927SAndroid Build Coastguard Worker engine_versions);
797*61046927SAndroid Build Coastguard Worker }
798*61046927SAndroid Build Coastguard Worker }
799*61046927SAndroid Build Coastguard Worker }
800*61046927SAndroid Build Coastguard Worker
801*61046927SAndroid Build Coastguard Worker /** \brief Parse attributes of an option element. */
802*61046927SAndroid Build Coastguard Worker static void
parseOptConfAttr(struct OptConfData * data,const char ** attr)803*61046927SAndroid Build Coastguard Worker parseOptConfAttr(struct OptConfData *data, const char **attr)
804*61046927SAndroid Build Coastguard Worker {
805*61046927SAndroid Build Coastguard Worker uint32_t i;
806*61046927SAndroid Build Coastguard Worker const char *name = NULL, *value = NULL;
807*61046927SAndroid Build Coastguard Worker for (i = 0; attr[i]; i += 2) {
808*61046927SAndroid Build Coastguard Worker if (!strcmp(attr[i], "name")) name = attr[i+1];
809*61046927SAndroid Build Coastguard Worker else if (!strcmp(attr[i], "value")) value = attr[i+1];
810*61046927SAndroid Build Coastguard Worker else XML_WARNING("unknown option attribute: %s.", attr[i]);
811*61046927SAndroid Build Coastguard Worker }
812*61046927SAndroid Build Coastguard Worker if (!name) XML_WARNING1("name attribute missing in option.");
813*61046927SAndroid Build Coastguard Worker if (!value) XML_WARNING1("value attribute missing in option.");
814*61046927SAndroid Build Coastguard Worker if (name && value) {
815*61046927SAndroid Build Coastguard Worker driOptionCache *cache = data->cache;
816*61046927SAndroid Build Coastguard Worker uint32_t opt = findOption(cache, name);
817*61046927SAndroid Build Coastguard Worker if (cache->info[opt].name == NULL)
818*61046927SAndroid Build Coastguard Worker /* don't use XML_WARNING, drirc defines options for all drivers,
819*61046927SAndroid Build Coastguard Worker * but not all drivers support them */
820*61046927SAndroid Build Coastguard Worker return;
821*61046927SAndroid Build Coastguard Worker else if (getenv(cache->info[opt].name)) {
822*61046927SAndroid Build Coastguard Worker /* don't use XML_WARNING, we want the user to see this! */
823*61046927SAndroid Build Coastguard Worker if (be_verbose()) {
824*61046927SAndroid Build Coastguard Worker fprintf(stderr,
825*61046927SAndroid Build Coastguard Worker "ATTENTION: option value of option %s ignored.\n",
826*61046927SAndroid Build Coastguard Worker cache->info[opt].name);
827*61046927SAndroid Build Coastguard Worker }
828*61046927SAndroid Build Coastguard Worker } else if (!parseValue(&cache->values[opt], cache->info[opt].type, value))
829*61046927SAndroid Build Coastguard Worker XML_WARNING("illegal option value: %s.", value);
830*61046927SAndroid Build Coastguard Worker }
831*61046927SAndroid Build Coastguard Worker }
832*61046927SAndroid Build Coastguard Worker
833*61046927SAndroid Build Coastguard Worker #if WITH_XMLCONFIG
834*61046927SAndroid Build Coastguard Worker
835*61046927SAndroid Build Coastguard Worker /** \brief Elements in configuration files. */
836*61046927SAndroid Build Coastguard Worker enum OptConfElem {
837*61046927SAndroid Build Coastguard Worker OC_APPLICATION = 0, OC_DEVICE, OC_DRICONF, OC_ENGINE, OC_OPTION, OC_COUNT
838*61046927SAndroid Build Coastguard Worker };
839*61046927SAndroid Build Coastguard Worker static const char *OptConfElems[] = {
840*61046927SAndroid Build Coastguard Worker [OC_APPLICATION] = "application",
841*61046927SAndroid Build Coastguard Worker [OC_DEVICE] = "device",
842*61046927SAndroid Build Coastguard Worker [OC_DRICONF] = "driconf",
843*61046927SAndroid Build Coastguard Worker [OC_ENGINE] = "engine",
844*61046927SAndroid Build Coastguard Worker [OC_OPTION] = "option",
845*61046927SAndroid Build Coastguard Worker };
846*61046927SAndroid Build Coastguard Worker
compare(const void * a,const void * b)847*61046927SAndroid Build Coastguard Worker static int compare(const void *a, const void *b) {
848*61046927SAndroid Build Coastguard Worker return strcmp(*(char *const*)a, *(char *const*)b);
849*61046927SAndroid Build Coastguard Worker }
850*61046927SAndroid Build Coastguard Worker /** \brief Binary search in a string array. */
851*61046927SAndroid Build Coastguard Worker static uint32_t
bsearchStr(const char * name,const char * elems[],uint32_t count)852*61046927SAndroid Build Coastguard Worker bsearchStr(const char *name, const char *elems[], uint32_t count)
853*61046927SAndroid Build Coastguard Worker {
854*61046927SAndroid Build Coastguard Worker const char **found;
855*61046927SAndroid Build Coastguard Worker found = bsearch(&name, elems, count, sizeof(char *), compare);
856*61046927SAndroid Build Coastguard Worker if (found)
857*61046927SAndroid Build Coastguard Worker return found - elems;
858*61046927SAndroid Build Coastguard Worker else
859*61046927SAndroid Build Coastguard Worker return count;
860*61046927SAndroid Build Coastguard Worker }
861*61046927SAndroid Build Coastguard Worker
862*61046927SAndroid Build Coastguard Worker /** \brief Handler for start element events. */
863*61046927SAndroid Build Coastguard Worker static void
optConfStartElem(void * userData,const char * name,const char ** attr)864*61046927SAndroid Build Coastguard Worker optConfStartElem(void *userData, const char *name,
865*61046927SAndroid Build Coastguard Worker const char **attr)
866*61046927SAndroid Build Coastguard Worker {
867*61046927SAndroid Build Coastguard Worker struct OptConfData *data = (struct OptConfData *)userData;
868*61046927SAndroid Build Coastguard Worker enum OptConfElem elem = bsearchStr(name, OptConfElems, OC_COUNT);
869*61046927SAndroid Build Coastguard Worker switch (elem) {
870*61046927SAndroid Build Coastguard Worker case OC_DRICONF:
871*61046927SAndroid Build Coastguard Worker if (data->inDriConf)
872*61046927SAndroid Build Coastguard Worker XML_WARNING1("nested <driconf> elements.");
873*61046927SAndroid Build Coastguard Worker if (attr[0])
874*61046927SAndroid Build Coastguard Worker XML_WARNING1("attributes specified on <driconf> element.");
875*61046927SAndroid Build Coastguard Worker data->inDriConf++;
876*61046927SAndroid Build Coastguard Worker break;
877*61046927SAndroid Build Coastguard Worker case OC_DEVICE:
878*61046927SAndroid Build Coastguard Worker if (!data->inDriConf)
879*61046927SAndroid Build Coastguard Worker XML_WARNING1("<device> should be inside <driconf>.");
880*61046927SAndroid Build Coastguard Worker if (data->inDevice)
881*61046927SAndroid Build Coastguard Worker XML_WARNING1("nested <device> elements.");
882*61046927SAndroid Build Coastguard Worker data->inDevice++;
883*61046927SAndroid Build Coastguard Worker if (!data->ignoringDevice && !data->ignoringApp)
884*61046927SAndroid Build Coastguard Worker parseDeviceAttr(data, attr);
885*61046927SAndroid Build Coastguard Worker break;
886*61046927SAndroid Build Coastguard Worker case OC_APPLICATION:
887*61046927SAndroid Build Coastguard Worker if (!data->inDevice)
888*61046927SAndroid Build Coastguard Worker XML_WARNING1("<application> should be inside <device>.");
889*61046927SAndroid Build Coastguard Worker if (data->inApp)
890*61046927SAndroid Build Coastguard Worker XML_WARNING1("nested <application> or <engine> elements.");
891*61046927SAndroid Build Coastguard Worker data->inApp++;
892*61046927SAndroid Build Coastguard Worker if (!data->ignoringDevice && !data->ignoringApp)
893*61046927SAndroid Build Coastguard Worker parseAppAttr(data, attr);
894*61046927SAndroid Build Coastguard Worker break;
895*61046927SAndroid Build Coastguard Worker case OC_ENGINE:
896*61046927SAndroid Build Coastguard Worker if (!data->inDevice)
897*61046927SAndroid Build Coastguard Worker XML_WARNING1("<engine> should be inside <device>.");
898*61046927SAndroid Build Coastguard Worker if (data->inApp)
899*61046927SAndroid Build Coastguard Worker XML_WARNING1("nested <application> or <engine> elements.");
900*61046927SAndroid Build Coastguard Worker data->inApp++;
901*61046927SAndroid Build Coastguard Worker if (!data->ignoringDevice && !data->ignoringApp)
902*61046927SAndroid Build Coastguard Worker parseEngineAttr(data, attr);
903*61046927SAndroid Build Coastguard Worker break;
904*61046927SAndroid Build Coastguard Worker case OC_OPTION:
905*61046927SAndroid Build Coastguard Worker if (!data->inApp)
906*61046927SAndroid Build Coastguard Worker XML_WARNING1("<option> should be inside <application>.");
907*61046927SAndroid Build Coastguard Worker if (data->inOption)
908*61046927SAndroid Build Coastguard Worker XML_WARNING1("nested <option> elements.");
909*61046927SAndroid Build Coastguard Worker data->inOption++;
910*61046927SAndroid Build Coastguard Worker if (!data->ignoringDevice && !data->ignoringApp)
911*61046927SAndroid Build Coastguard Worker parseOptConfAttr(data, attr);
912*61046927SAndroid Build Coastguard Worker break;
913*61046927SAndroid Build Coastguard Worker default:
914*61046927SAndroid Build Coastguard Worker XML_WARNING("unknown element: %s.", name);
915*61046927SAndroid Build Coastguard Worker }
916*61046927SAndroid Build Coastguard Worker }
917*61046927SAndroid Build Coastguard Worker
918*61046927SAndroid Build Coastguard Worker /** \brief Handler for end element events. */
919*61046927SAndroid Build Coastguard Worker static void
optConfEndElem(void * userData,const char * name)920*61046927SAndroid Build Coastguard Worker optConfEndElem(void *userData, const char *name)
921*61046927SAndroid Build Coastguard Worker {
922*61046927SAndroid Build Coastguard Worker struct OptConfData *data = (struct OptConfData *)userData;
923*61046927SAndroid Build Coastguard Worker enum OptConfElem elem = bsearchStr(name, OptConfElems, OC_COUNT);
924*61046927SAndroid Build Coastguard Worker switch (elem) {
925*61046927SAndroid Build Coastguard Worker case OC_DRICONF:
926*61046927SAndroid Build Coastguard Worker data->inDriConf--;
927*61046927SAndroid Build Coastguard Worker break;
928*61046927SAndroid Build Coastguard Worker case OC_DEVICE:
929*61046927SAndroid Build Coastguard Worker if (data->inDevice-- == data->ignoringDevice)
930*61046927SAndroid Build Coastguard Worker data->ignoringDevice = 0;
931*61046927SAndroid Build Coastguard Worker break;
932*61046927SAndroid Build Coastguard Worker case OC_APPLICATION:
933*61046927SAndroid Build Coastguard Worker case OC_ENGINE:
934*61046927SAndroid Build Coastguard Worker if (data->inApp-- == data->ignoringApp)
935*61046927SAndroid Build Coastguard Worker data->ignoringApp = 0;
936*61046927SAndroid Build Coastguard Worker break;
937*61046927SAndroid Build Coastguard Worker case OC_OPTION:
938*61046927SAndroid Build Coastguard Worker data->inOption--;
939*61046927SAndroid Build Coastguard Worker break;
940*61046927SAndroid Build Coastguard Worker default:
941*61046927SAndroid Build Coastguard Worker /* unknown element, warning was produced on start tag */;
942*61046927SAndroid Build Coastguard Worker }
943*61046927SAndroid Build Coastguard Worker }
944*61046927SAndroid Build Coastguard Worker
945*61046927SAndroid Build Coastguard Worker static void
_parseOneConfigFile(XML_Parser p)946*61046927SAndroid Build Coastguard Worker _parseOneConfigFile(XML_Parser p)
947*61046927SAndroid Build Coastguard Worker {
948*61046927SAndroid Build Coastguard Worker #define BUF_SIZE 0x1000
949*61046927SAndroid Build Coastguard Worker struct OptConfData *data = (struct OptConfData *)XML_GetUserData(p);
950*61046927SAndroid Build Coastguard Worker int status;
951*61046927SAndroid Build Coastguard Worker int fd;
952*61046927SAndroid Build Coastguard Worker
953*61046927SAndroid Build Coastguard Worker if ((fd = open(data->name, O_RDONLY)) == -1) {
954*61046927SAndroid Build Coastguard Worker __driUtilMessage("Can't open configuration file %s: %s.",
955*61046927SAndroid Build Coastguard Worker data->name, strerror(errno));
956*61046927SAndroid Build Coastguard Worker return;
957*61046927SAndroid Build Coastguard Worker }
958*61046927SAndroid Build Coastguard Worker
959*61046927SAndroid Build Coastguard Worker while (1) {
960*61046927SAndroid Build Coastguard Worker int bytesRead;
961*61046927SAndroid Build Coastguard Worker void *buffer = XML_GetBuffer(p, BUF_SIZE);
962*61046927SAndroid Build Coastguard Worker if (!buffer) {
963*61046927SAndroid Build Coastguard Worker __driUtilMessage("Can't allocate parser buffer.");
964*61046927SAndroid Build Coastguard Worker break;
965*61046927SAndroid Build Coastguard Worker }
966*61046927SAndroid Build Coastguard Worker bytesRead = read(fd, buffer, BUF_SIZE);
967*61046927SAndroid Build Coastguard Worker if (bytesRead == -1) {
968*61046927SAndroid Build Coastguard Worker __driUtilMessage("Error reading from configuration file %s: %s.",
969*61046927SAndroid Build Coastguard Worker data->name, strerror(errno));
970*61046927SAndroid Build Coastguard Worker break;
971*61046927SAndroid Build Coastguard Worker }
972*61046927SAndroid Build Coastguard Worker status = XML_ParseBuffer(p, bytesRead, bytesRead == 0);
973*61046927SAndroid Build Coastguard Worker if (!status) {
974*61046927SAndroid Build Coastguard Worker XML_ERROR("%s.", XML_ErrorString(XML_GetErrorCode(p)));
975*61046927SAndroid Build Coastguard Worker break;
976*61046927SAndroid Build Coastguard Worker }
977*61046927SAndroid Build Coastguard Worker if (bytesRead == 0)
978*61046927SAndroid Build Coastguard Worker break;
979*61046927SAndroid Build Coastguard Worker }
980*61046927SAndroid Build Coastguard Worker
981*61046927SAndroid Build Coastguard Worker close(fd);
982*61046927SAndroid Build Coastguard Worker #undef BUF_SIZE
983*61046927SAndroid Build Coastguard Worker }
984*61046927SAndroid Build Coastguard Worker
985*61046927SAndroid Build Coastguard Worker /** \brief Parse the named configuration file */
986*61046927SAndroid Build Coastguard Worker static void
parseOneConfigFile(struct OptConfData * data,const char * filename)987*61046927SAndroid Build Coastguard Worker parseOneConfigFile(struct OptConfData *data, const char *filename)
988*61046927SAndroid Build Coastguard Worker {
989*61046927SAndroid Build Coastguard Worker XML_Parser p;
990*61046927SAndroid Build Coastguard Worker
991*61046927SAndroid Build Coastguard Worker p = XML_ParserCreate(NULL); /* use encoding specified by file */
992*61046927SAndroid Build Coastguard Worker XML_SetElementHandler(p, optConfStartElem, optConfEndElem);
993*61046927SAndroid Build Coastguard Worker XML_SetUserData(p, data);
994*61046927SAndroid Build Coastguard Worker data->parser = p;
995*61046927SAndroid Build Coastguard Worker data->name = filename;
996*61046927SAndroid Build Coastguard Worker data->ignoringDevice = 0;
997*61046927SAndroid Build Coastguard Worker data->ignoringApp = 0;
998*61046927SAndroid Build Coastguard Worker data->inDriConf = 0;
999*61046927SAndroid Build Coastguard Worker data->inDevice = 0;
1000*61046927SAndroid Build Coastguard Worker data->inApp = 0;
1001*61046927SAndroid Build Coastguard Worker data->inOption = 0;
1002*61046927SAndroid Build Coastguard Worker
1003*61046927SAndroid Build Coastguard Worker _parseOneConfigFile(p);
1004*61046927SAndroid Build Coastguard Worker XML_ParserFree(p);
1005*61046927SAndroid Build Coastguard Worker }
1006*61046927SAndroid Build Coastguard Worker
1007*61046927SAndroid Build Coastguard Worker static int
scandir_filter(const struct dirent * ent)1008*61046927SAndroid Build Coastguard Worker scandir_filter(const struct dirent *ent)
1009*61046927SAndroid Build Coastguard Worker {
1010*61046927SAndroid Build Coastguard Worker #ifndef DT_REG /* systems without d_type in dirent results */
1011*61046927SAndroid Build Coastguard Worker struct stat st;
1012*61046927SAndroid Build Coastguard Worker
1013*61046927SAndroid Build Coastguard Worker if ((lstat(ent->d_name, &st) != 0) ||
1014*61046927SAndroid Build Coastguard Worker (!S_ISREG(st.st_mode) && !S_ISLNK(st.st_mode)))
1015*61046927SAndroid Build Coastguard Worker return 0;
1016*61046927SAndroid Build Coastguard Worker #else
1017*61046927SAndroid Build Coastguard Worker /* Allow through unknown file types for filesystems that don't support d_type
1018*61046927SAndroid Build Coastguard Worker * The full filepath isn't available here to stat the file
1019*61046927SAndroid Build Coastguard Worker */
1020*61046927SAndroid Build Coastguard Worker if (ent->d_type != DT_REG && ent->d_type != DT_LNK && ent->d_type != DT_UNKNOWN)
1021*61046927SAndroid Build Coastguard Worker return 0;
1022*61046927SAndroid Build Coastguard Worker #endif
1023*61046927SAndroid Build Coastguard Worker
1024*61046927SAndroid Build Coastguard Worker int len = strlen(ent->d_name);
1025*61046927SAndroid Build Coastguard Worker if (len <= 5 || strcmp(ent->d_name + len - 5, ".conf"))
1026*61046927SAndroid Build Coastguard Worker return 0;
1027*61046927SAndroid Build Coastguard Worker
1028*61046927SAndroid Build Coastguard Worker return 1;
1029*61046927SAndroid Build Coastguard Worker }
1030*61046927SAndroid Build Coastguard Worker
1031*61046927SAndroid Build Coastguard Worker /** \brief Parse configuration files in a directory */
1032*61046927SAndroid Build Coastguard Worker static void
parseConfigDir(struct OptConfData * data,const char * dirname)1033*61046927SAndroid Build Coastguard Worker parseConfigDir(struct OptConfData *data, const char *dirname)
1034*61046927SAndroid Build Coastguard Worker {
1035*61046927SAndroid Build Coastguard Worker int i, count;
1036*61046927SAndroid Build Coastguard Worker struct dirent **entries = NULL;
1037*61046927SAndroid Build Coastguard Worker
1038*61046927SAndroid Build Coastguard Worker count = scandir(dirname, &entries, scandir_filter, alphasort);
1039*61046927SAndroid Build Coastguard Worker if (count < 0)
1040*61046927SAndroid Build Coastguard Worker return;
1041*61046927SAndroid Build Coastguard Worker
1042*61046927SAndroid Build Coastguard Worker for (i = 0; i < count; i++) {
1043*61046927SAndroid Build Coastguard Worker char filename[PATH_MAX];
1044*61046927SAndroid Build Coastguard Worker #ifdef DT_REG
1045*61046927SAndroid Build Coastguard Worker unsigned char d_type = entries[i]->d_type;
1046*61046927SAndroid Build Coastguard Worker #endif
1047*61046927SAndroid Build Coastguard Worker
1048*61046927SAndroid Build Coastguard Worker snprintf(filename, PATH_MAX, "%s/%s", dirname, entries[i]->d_name);
1049*61046927SAndroid Build Coastguard Worker free(entries[i]);
1050*61046927SAndroid Build Coastguard Worker
1051*61046927SAndroid Build Coastguard Worker #ifdef DT_REG
1052*61046927SAndroid Build Coastguard Worker /* In the case of unknown d_type, ensure it is a regular file
1053*61046927SAndroid Build Coastguard Worker * This can be accomplished with stat on the full filepath
1054*61046927SAndroid Build Coastguard Worker */
1055*61046927SAndroid Build Coastguard Worker if (d_type == DT_UNKNOWN) {
1056*61046927SAndroid Build Coastguard Worker struct stat st;
1057*61046927SAndroid Build Coastguard Worker if (stat(filename, &st) != 0 ||
1058*61046927SAndroid Build Coastguard Worker !S_ISREG(st.st_mode)) {
1059*61046927SAndroid Build Coastguard Worker continue;
1060*61046927SAndroid Build Coastguard Worker }
1061*61046927SAndroid Build Coastguard Worker }
1062*61046927SAndroid Build Coastguard Worker #endif
1063*61046927SAndroid Build Coastguard Worker
1064*61046927SAndroid Build Coastguard Worker parseOneConfigFile(data, filename);
1065*61046927SAndroid Build Coastguard Worker }
1066*61046927SAndroid Build Coastguard Worker
1067*61046927SAndroid Build Coastguard Worker free(entries);
1068*61046927SAndroid Build Coastguard Worker }
1069*61046927SAndroid Build Coastguard Worker #else
1070*61046927SAndroid Build Coastguard Worker # include "driconf_static.h"
1071*61046927SAndroid Build Coastguard Worker
1072*61046927SAndroid Build Coastguard Worker static void
parseStaticOptions(struct OptConfData * data,const struct driconf_option * options,unsigned num_options)1073*61046927SAndroid Build Coastguard Worker parseStaticOptions(struct OptConfData *data, const struct driconf_option *options,
1074*61046927SAndroid Build Coastguard Worker unsigned num_options)
1075*61046927SAndroid Build Coastguard Worker {
1076*61046927SAndroid Build Coastguard Worker if (data->ignoringDevice || data->ignoringApp)
1077*61046927SAndroid Build Coastguard Worker return;
1078*61046927SAndroid Build Coastguard Worker for (unsigned i = 0; i < num_options; i++) {
1079*61046927SAndroid Build Coastguard Worker const char *optattr[] = {
1080*61046927SAndroid Build Coastguard Worker "name", options[i].name,
1081*61046927SAndroid Build Coastguard Worker "value", options[i].value,
1082*61046927SAndroid Build Coastguard Worker NULL
1083*61046927SAndroid Build Coastguard Worker };
1084*61046927SAndroid Build Coastguard Worker parseOptConfAttr(data, optattr);
1085*61046927SAndroid Build Coastguard Worker }
1086*61046927SAndroid Build Coastguard Worker }
1087*61046927SAndroid Build Coastguard Worker
1088*61046927SAndroid Build Coastguard Worker static void
parseStaticConfig(struct OptConfData * data)1089*61046927SAndroid Build Coastguard Worker parseStaticConfig(struct OptConfData *data)
1090*61046927SAndroid Build Coastguard Worker {
1091*61046927SAndroid Build Coastguard Worker data->ignoringDevice = 0;
1092*61046927SAndroid Build Coastguard Worker data->ignoringApp = 0;
1093*61046927SAndroid Build Coastguard Worker data->inDriConf = 0;
1094*61046927SAndroid Build Coastguard Worker data->inDevice = 0;
1095*61046927SAndroid Build Coastguard Worker data->inApp = 0;
1096*61046927SAndroid Build Coastguard Worker data->inOption = 0;
1097*61046927SAndroid Build Coastguard Worker
1098*61046927SAndroid Build Coastguard Worker for (unsigned i = 0; i < ARRAY_SIZE(driconf); i++) {
1099*61046927SAndroid Build Coastguard Worker const struct driconf_device *d = driconf[i];
1100*61046927SAndroid Build Coastguard Worker const char *devattr[] = {
1101*61046927SAndroid Build Coastguard Worker "driver", d->driver,
1102*61046927SAndroid Build Coastguard Worker "device", d->device,
1103*61046927SAndroid Build Coastguard Worker NULL
1104*61046927SAndroid Build Coastguard Worker };
1105*61046927SAndroid Build Coastguard Worker
1106*61046927SAndroid Build Coastguard Worker data->ignoringDevice = 0;
1107*61046927SAndroid Build Coastguard Worker data->inDevice++;
1108*61046927SAndroid Build Coastguard Worker parseDeviceAttr(data, devattr);
1109*61046927SAndroid Build Coastguard Worker data->inDevice--;
1110*61046927SAndroid Build Coastguard Worker
1111*61046927SAndroid Build Coastguard Worker data->inApp++;
1112*61046927SAndroid Build Coastguard Worker
1113*61046927SAndroid Build Coastguard Worker for (unsigned j = 0; j < d->num_engines; j++) {
1114*61046927SAndroid Build Coastguard Worker const struct driconf_engine *e = &d->engines[j];
1115*61046927SAndroid Build Coastguard Worker const char *engattr[] = {
1116*61046927SAndroid Build Coastguard Worker "engine_name_match", e->engine_name_match,
1117*61046927SAndroid Build Coastguard Worker "engine_versions", e->engine_versions,
1118*61046927SAndroid Build Coastguard Worker NULL
1119*61046927SAndroid Build Coastguard Worker };
1120*61046927SAndroid Build Coastguard Worker
1121*61046927SAndroid Build Coastguard Worker data->ignoringApp = 0;
1122*61046927SAndroid Build Coastguard Worker parseEngineAttr(data, engattr);
1123*61046927SAndroid Build Coastguard Worker parseStaticOptions(data, e->options, e->num_options);
1124*61046927SAndroid Build Coastguard Worker }
1125*61046927SAndroid Build Coastguard Worker
1126*61046927SAndroid Build Coastguard Worker for (unsigned j = 0; j < d->num_applications; j++) {
1127*61046927SAndroid Build Coastguard Worker const struct driconf_application *a = &d->applications[j];
1128*61046927SAndroid Build Coastguard Worker const char *appattr[] = {
1129*61046927SAndroid Build Coastguard Worker "name", a->name,
1130*61046927SAndroid Build Coastguard Worker "executable", a->executable,
1131*61046927SAndroid Build Coastguard Worker "executable_regexp", a->executable_regexp,
1132*61046927SAndroid Build Coastguard Worker "sha1", a->sha1,
1133*61046927SAndroid Build Coastguard Worker "application_name_match", a->application_name_match,
1134*61046927SAndroid Build Coastguard Worker "application_versions", a->application_versions,
1135*61046927SAndroid Build Coastguard Worker NULL
1136*61046927SAndroid Build Coastguard Worker };
1137*61046927SAndroid Build Coastguard Worker
1138*61046927SAndroid Build Coastguard Worker data->ignoringApp = 0;
1139*61046927SAndroid Build Coastguard Worker parseAppAttr(data, appattr);
1140*61046927SAndroid Build Coastguard Worker parseStaticOptions(data, a->options, a->num_options);
1141*61046927SAndroid Build Coastguard Worker }
1142*61046927SAndroid Build Coastguard Worker
1143*61046927SAndroid Build Coastguard Worker data->inApp--;
1144*61046927SAndroid Build Coastguard Worker }
1145*61046927SAndroid Build Coastguard Worker }
1146*61046927SAndroid Build Coastguard Worker #endif /* WITH_XMLCONFIG */
1147*61046927SAndroid Build Coastguard Worker
1148*61046927SAndroid Build Coastguard Worker /** \brief Initialize an option cache based on info */
1149*61046927SAndroid Build Coastguard Worker static void
initOptionCache(driOptionCache * cache,const driOptionCache * info)1150*61046927SAndroid Build Coastguard Worker initOptionCache(driOptionCache *cache, const driOptionCache *info)
1151*61046927SAndroid Build Coastguard Worker {
1152*61046927SAndroid Build Coastguard Worker unsigned i, size = 1 << info->tableSize;
1153*61046927SAndroid Build Coastguard Worker cache->info = info->info;
1154*61046927SAndroid Build Coastguard Worker cache->tableSize = info->tableSize;
1155*61046927SAndroid Build Coastguard Worker cache->values = malloc(((size_t)1 << info->tableSize) * sizeof(driOptionValue));
1156*61046927SAndroid Build Coastguard Worker if (cache->values == NULL) {
1157*61046927SAndroid Build Coastguard Worker fprintf(stderr, "%s: %d: out of memory.\n", __FILE__, __LINE__);
1158*61046927SAndroid Build Coastguard Worker abort();
1159*61046927SAndroid Build Coastguard Worker }
1160*61046927SAndroid Build Coastguard Worker memcpy(cache->values, info->values,
1161*61046927SAndroid Build Coastguard Worker ((size_t)1 << info->tableSize) * sizeof(driOptionValue));
1162*61046927SAndroid Build Coastguard Worker for (i = 0; i < size; ++i) {
1163*61046927SAndroid Build Coastguard Worker if (cache->info[i].type == DRI_STRING)
1164*61046927SAndroid Build Coastguard Worker XSTRDUP(cache->values[i]._string, info->values[i]._string);
1165*61046927SAndroid Build Coastguard Worker }
1166*61046927SAndroid Build Coastguard Worker }
1167*61046927SAndroid Build Coastguard Worker
1168*61046927SAndroid Build Coastguard Worker static const char *execname;
1169*61046927SAndroid Build Coastguard Worker
1170*61046927SAndroid Build Coastguard Worker void
driInjectExecName(const char * exec)1171*61046927SAndroid Build Coastguard Worker driInjectExecName(const char *exec)
1172*61046927SAndroid Build Coastguard Worker {
1173*61046927SAndroid Build Coastguard Worker execname = exec;
1174*61046927SAndroid Build Coastguard Worker }
1175*61046927SAndroid Build Coastguard Worker
1176*61046927SAndroid Build Coastguard Worker void
driParseConfigFiles(driOptionCache * cache,const driOptionCache * info,int screenNum,const char * driverName,const char * kernelDriverName,const char * deviceName,const char * applicationName,uint32_t applicationVersion,const char * engineName,uint32_t engineVersion)1177*61046927SAndroid Build Coastguard Worker driParseConfigFiles(driOptionCache *cache, const driOptionCache *info,
1178*61046927SAndroid Build Coastguard Worker int screenNum, const char *driverName,
1179*61046927SAndroid Build Coastguard Worker const char *kernelDriverName,
1180*61046927SAndroid Build Coastguard Worker const char *deviceName,
1181*61046927SAndroid Build Coastguard Worker const char *applicationName, uint32_t applicationVersion,
1182*61046927SAndroid Build Coastguard Worker const char *engineName, uint32_t engineVersion)
1183*61046927SAndroid Build Coastguard Worker {
1184*61046927SAndroid Build Coastguard Worker initOptionCache(cache, info);
1185*61046927SAndroid Build Coastguard Worker struct OptConfData userData = {0};
1186*61046927SAndroid Build Coastguard Worker
1187*61046927SAndroid Build Coastguard Worker if (!execname)
1188*61046927SAndroid Build Coastguard Worker execname = os_get_option("MESA_DRICONF_EXECUTABLE_OVERRIDE");
1189*61046927SAndroid Build Coastguard Worker if (!execname)
1190*61046927SAndroid Build Coastguard Worker execname = util_get_process_name();
1191*61046927SAndroid Build Coastguard Worker
1192*61046927SAndroid Build Coastguard Worker userData.cache = cache;
1193*61046927SAndroid Build Coastguard Worker userData.screenNum = screenNum;
1194*61046927SAndroid Build Coastguard Worker userData.driverName = driverName;
1195*61046927SAndroid Build Coastguard Worker userData.kernelDriverName = kernelDriverName;
1196*61046927SAndroid Build Coastguard Worker userData.deviceName = deviceName;
1197*61046927SAndroid Build Coastguard Worker userData.applicationName = applicationName ? applicationName : "";
1198*61046927SAndroid Build Coastguard Worker userData.applicationVersion = applicationVersion;
1199*61046927SAndroid Build Coastguard Worker userData.engineName = engineName ? engineName : "";
1200*61046927SAndroid Build Coastguard Worker userData.engineVersion = engineVersion;
1201*61046927SAndroid Build Coastguard Worker userData.execName = execname;
1202*61046927SAndroid Build Coastguard Worker
1203*61046927SAndroid Build Coastguard Worker #if WITH_XMLCONFIG
1204*61046927SAndroid Build Coastguard Worker char *home, *configdir;
1205*61046927SAndroid Build Coastguard Worker
1206*61046927SAndroid Build Coastguard Worker /* parse from either $DRIRC_CONFIGDIR or $datadir/drirc.d */
1207*61046927SAndroid Build Coastguard Worker if ((configdir = getenv("DRIRC_CONFIGDIR")))
1208*61046927SAndroid Build Coastguard Worker parseConfigDir(&userData, configdir);
1209*61046927SAndroid Build Coastguard Worker else {
1210*61046927SAndroid Build Coastguard Worker parseConfigDir(&userData, DATADIR "/drirc.d");
1211*61046927SAndroid Build Coastguard Worker parseOneConfigFile(&userData, SYSCONFDIR "/drirc");
1212*61046927SAndroid Build Coastguard Worker }
1213*61046927SAndroid Build Coastguard Worker
1214*61046927SAndroid Build Coastguard Worker if ((home = getenv("HOME"))) {
1215*61046927SAndroid Build Coastguard Worker char filename[PATH_MAX];
1216*61046927SAndroid Build Coastguard Worker
1217*61046927SAndroid Build Coastguard Worker snprintf(filename, PATH_MAX, "%s/.drirc", home);
1218*61046927SAndroid Build Coastguard Worker parseOneConfigFile(&userData, filename);
1219*61046927SAndroid Build Coastguard Worker }
1220*61046927SAndroid Build Coastguard Worker #else
1221*61046927SAndroid Build Coastguard Worker parseStaticConfig(&userData);
1222*61046927SAndroid Build Coastguard Worker #endif /* WITH_XMLCONFIG */
1223*61046927SAndroid Build Coastguard Worker }
1224*61046927SAndroid Build Coastguard Worker
1225*61046927SAndroid Build Coastguard Worker void
driDestroyOptionInfo(driOptionCache * info)1226*61046927SAndroid Build Coastguard Worker driDestroyOptionInfo(driOptionCache *info)
1227*61046927SAndroid Build Coastguard Worker {
1228*61046927SAndroid Build Coastguard Worker driDestroyOptionCache(info);
1229*61046927SAndroid Build Coastguard Worker if (info->info) {
1230*61046927SAndroid Build Coastguard Worker uint32_t i, size = 1 << info->tableSize;
1231*61046927SAndroid Build Coastguard Worker for (i = 0; i < size; ++i) {
1232*61046927SAndroid Build Coastguard Worker if (info->info[i].name) {
1233*61046927SAndroid Build Coastguard Worker free(info->info[i].name);
1234*61046927SAndroid Build Coastguard Worker }
1235*61046927SAndroid Build Coastguard Worker }
1236*61046927SAndroid Build Coastguard Worker free(info->info);
1237*61046927SAndroid Build Coastguard Worker }
1238*61046927SAndroid Build Coastguard Worker }
1239*61046927SAndroid Build Coastguard Worker
1240*61046927SAndroid Build Coastguard Worker void
driDestroyOptionCache(driOptionCache * cache)1241*61046927SAndroid Build Coastguard Worker driDestroyOptionCache(driOptionCache *cache)
1242*61046927SAndroid Build Coastguard Worker {
1243*61046927SAndroid Build Coastguard Worker if (cache->info) {
1244*61046927SAndroid Build Coastguard Worker unsigned i, size = 1 << cache->tableSize;
1245*61046927SAndroid Build Coastguard Worker for (i = 0; i < size; ++i) {
1246*61046927SAndroid Build Coastguard Worker if (cache->info[i].type == DRI_STRING)
1247*61046927SAndroid Build Coastguard Worker free(cache->values[i]._string);
1248*61046927SAndroid Build Coastguard Worker }
1249*61046927SAndroid Build Coastguard Worker }
1250*61046927SAndroid Build Coastguard Worker free(cache->values);
1251*61046927SAndroid Build Coastguard Worker }
1252*61046927SAndroid Build Coastguard Worker
1253*61046927SAndroid Build Coastguard Worker unsigned char
driCheckOption(const driOptionCache * cache,const char * name,driOptionType type)1254*61046927SAndroid Build Coastguard Worker driCheckOption(const driOptionCache *cache, const char *name,
1255*61046927SAndroid Build Coastguard Worker driOptionType type)
1256*61046927SAndroid Build Coastguard Worker {
1257*61046927SAndroid Build Coastguard Worker uint32_t i = findOption(cache, name);
1258*61046927SAndroid Build Coastguard Worker return cache->info[i].name != NULL && cache->info[i].type == type;
1259*61046927SAndroid Build Coastguard Worker }
1260*61046927SAndroid Build Coastguard Worker
1261*61046927SAndroid Build Coastguard Worker unsigned char
driQueryOptionb(const driOptionCache * cache,const char * name)1262*61046927SAndroid Build Coastguard Worker driQueryOptionb(const driOptionCache *cache, const char *name)
1263*61046927SAndroid Build Coastguard Worker {
1264*61046927SAndroid Build Coastguard Worker uint32_t i = findOption(cache, name);
1265*61046927SAndroid Build Coastguard Worker /* make sure the option is defined and has the correct type */
1266*61046927SAndroid Build Coastguard Worker assert(cache->info[i].name != NULL);
1267*61046927SAndroid Build Coastguard Worker assert(cache->info[i].type == DRI_BOOL);
1268*61046927SAndroid Build Coastguard Worker return cache->values[i]._bool;
1269*61046927SAndroid Build Coastguard Worker }
1270*61046927SAndroid Build Coastguard Worker
1271*61046927SAndroid Build Coastguard Worker int
driQueryOptioni(const driOptionCache * cache,const char * name)1272*61046927SAndroid Build Coastguard Worker driQueryOptioni(const driOptionCache *cache, const char *name)
1273*61046927SAndroid Build Coastguard Worker {
1274*61046927SAndroid Build Coastguard Worker uint32_t i = findOption(cache, name);
1275*61046927SAndroid Build Coastguard Worker /* make sure the option is defined and has the correct type */
1276*61046927SAndroid Build Coastguard Worker assert(cache->info[i].name != NULL);
1277*61046927SAndroid Build Coastguard Worker assert(cache->info[i].type == DRI_INT || cache->info[i].type == DRI_ENUM);
1278*61046927SAndroid Build Coastguard Worker return cache->values[i]._int;
1279*61046927SAndroid Build Coastguard Worker }
1280*61046927SAndroid Build Coastguard Worker
1281*61046927SAndroid Build Coastguard Worker float
driQueryOptionf(const driOptionCache * cache,const char * name)1282*61046927SAndroid Build Coastguard Worker driQueryOptionf(const driOptionCache *cache, const char *name)
1283*61046927SAndroid Build Coastguard Worker {
1284*61046927SAndroid Build Coastguard Worker uint32_t i = findOption(cache, name);
1285*61046927SAndroid Build Coastguard Worker /* make sure the option is defined and has the correct type */
1286*61046927SAndroid Build Coastguard Worker assert(cache->info[i].name != NULL);
1287*61046927SAndroid Build Coastguard Worker assert(cache->info[i].type == DRI_FLOAT);
1288*61046927SAndroid Build Coastguard Worker return cache->values[i]._float;
1289*61046927SAndroid Build Coastguard Worker }
1290*61046927SAndroid Build Coastguard Worker
1291*61046927SAndroid Build Coastguard Worker char *
driQueryOptionstr(const driOptionCache * cache,const char * name)1292*61046927SAndroid Build Coastguard Worker driQueryOptionstr(const driOptionCache *cache, const char *name)
1293*61046927SAndroid Build Coastguard Worker {
1294*61046927SAndroid Build Coastguard Worker uint32_t i = findOption(cache, name);
1295*61046927SAndroid Build Coastguard Worker /* make sure the option is defined and has the correct type */
1296*61046927SAndroid Build Coastguard Worker assert(cache->info[i].name != NULL);
1297*61046927SAndroid Build Coastguard Worker assert(cache->info[i].type == DRI_STRING);
1298*61046927SAndroid Build Coastguard Worker return cache->values[i]._string;
1299*61046927SAndroid Build Coastguard Worker }
1300