1355b4669Sjacobs /*
2355b4669Sjacobs * CDDL HEADER START
3355b4669Sjacobs *
4355b4669Sjacobs * The contents of this file are subject to the terms of the
5355b4669Sjacobs * Common Development and Distribution License (the "License").
6355b4669Sjacobs * You may not use this file except in compliance with the License.
7355b4669Sjacobs *
8355b4669Sjacobs * You can obtain a copy of the license at usr/src/OPENSOLARIS.LICENSE
9355b4669Sjacobs * or http://www.opensolaris.org/os/licensing.
10355b4669Sjacobs * See the License for the specific language governing permissions
11355b4669Sjacobs * and limitations under the License.
12355b4669Sjacobs *
13355b4669Sjacobs * When distributing Covered Code, include this CDDL HEADER in each
14355b4669Sjacobs * file and include the License file at usr/src/OPENSOLARIS.LICENSE.
15355b4669Sjacobs * If applicable, add the following below this CDDL HEADER, with the
16355b4669Sjacobs * fields enclosed by brackets "[]" replaced with your own identifying
17355b4669Sjacobs * information: Portions Copyright [yyyy] [name of copyright owner]
18355b4669Sjacobs *
19355b4669Sjacobs * CDDL HEADER END
20355b4669Sjacobs */
21355b4669Sjacobs
22355b4669Sjacobs /*
23ee169c7eSGary Mills * Copyright (c) 2014 Gary Mills
24ee169c7eSGary Mills *
2543b9c050Sjacobs * Copyright 2008 Sun Microsystems, Inc. All rights reserved.
26355b4669Sjacobs * Use is subject to license terms.
27355b4669Sjacobs *
28355b4669Sjacobs */
29355b4669Sjacobs
30355b4669Sjacobs /*LINTLIBRARY*/
31355b4669Sjacobs
32355b4669Sjacobs #include <stdio.h>
33355b4669Sjacobs #include <stdlib.h>
34355b4669Sjacobs #include <stdarg.h>
35355b4669Sjacobs #include <string.h>
3643b9c050Sjacobs #include <ctype.h>
37355b4669Sjacobs #include <alloca.h>
38355b4669Sjacobs #include <papi.h>
3943b9c050Sjacobs #include <regex.h>
40355b4669Sjacobs
41fa9d6f9fSsg223391 #define MAX_PAGES 32767
42fa9d6f9fSsg223391 /*
43fa9d6f9fSsg223391 * Assuming the maximum number of pages in
44fa9d6f9fSsg223391 * a document to be 32767
45fa9d6f9fSsg223391 */
46fa9d6f9fSsg223391
47355b4669Sjacobs static void papiAttributeFree(papi_attribute_t *attribute);
48355b4669Sjacobs
49355b4669Sjacobs static void
papiAttributeValueFree(papi_attribute_value_type_t type,papi_attribute_value_t * value)50355b4669Sjacobs papiAttributeValueFree(papi_attribute_value_type_t type,
51355b4669Sjacobs papi_attribute_value_t *value)
52355b4669Sjacobs {
53355b4669Sjacobs if (value != NULL) {
54355b4669Sjacobs switch (type) {
55355b4669Sjacobs case PAPI_STRING:
56355b4669Sjacobs if (value->string != NULL)
57355b4669Sjacobs free(value->string);
58355b4669Sjacobs break;
59355b4669Sjacobs case PAPI_COLLECTION:
60355b4669Sjacobs if (value->collection != NULL) {
61355b4669Sjacobs int i;
62355b4669Sjacobs
63355b4669Sjacobs for (i = 0; value->collection[i] != NULL; i++)
64355b4669Sjacobs papiAttributeFree(value->collection[i]);
65355b4669Sjacobs
66355b4669Sjacobs free(value->collection);
67355b4669Sjacobs }
68355b4669Sjacobs break;
69355b4669Sjacobs default: /* don't need to free anything extra */
70355b4669Sjacobs break;
71355b4669Sjacobs }
72355b4669Sjacobs
73355b4669Sjacobs free(value);
74355b4669Sjacobs }
75355b4669Sjacobs }
76355b4669Sjacobs
77355b4669Sjacobs static void
papiAttributeValuesFree(papi_attribute_value_type_t type,papi_attribute_value_t ** values)78355b4669Sjacobs papiAttributeValuesFree(papi_attribute_value_type_t type,
79355b4669Sjacobs papi_attribute_value_t **values)
80355b4669Sjacobs {
81355b4669Sjacobs if (values != NULL) {
82355b4669Sjacobs int i;
83355b4669Sjacobs
84355b4669Sjacobs for (i = 0; values[i] != NULL; i++)
85355b4669Sjacobs papiAttributeValueFree(type, values[i]);
86355b4669Sjacobs
87355b4669Sjacobs free(values);
88355b4669Sjacobs }
89355b4669Sjacobs }
90355b4669Sjacobs
91355b4669Sjacobs static void
papiAttributeFree(papi_attribute_t * attribute)92355b4669Sjacobs papiAttributeFree(papi_attribute_t *attribute)
93355b4669Sjacobs {
94355b4669Sjacobs if (attribute != NULL) {
95355b4669Sjacobs free(attribute->name);
96355b4669Sjacobs if (attribute->values != NULL)
97355b4669Sjacobs papiAttributeValuesFree(attribute->type,
98355b4669Sjacobs attribute->values);
99355b4669Sjacobs free(attribute);
100355b4669Sjacobs }
101355b4669Sjacobs }
102355b4669Sjacobs
103355b4669Sjacobs void
papiAttributeListFree(papi_attribute_t ** list)104355b4669Sjacobs papiAttributeListFree(papi_attribute_t **list)
105355b4669Sjacobs {
106355b4669Sjacobs if (list != NULL) {
107355b4669Sjacobs int i;
108355b4669Sjacobs
109355b4669Sjacobs for (i = 0; list[i] != NULL; i++)
110355b4669Sjacobs papiAttributeFree(list[i]);
111355b4669Sjacobs
112355b4669Sjacobs free(list);
113355b4669Sjacobs }
114355b4669Sjacobs }
115355b4669Sjacobs
116355b4669Sjacobs static papi_attribute_t **
collection_dup(papi_attribute_t ** collection)117355b4669Sjacobs collection_dup(papi_attribute_t **collection)
118355b4669Sjacobs {
119355b4669Sjacobs papi_attribute_t **result = NULL;
120355b4669Sjacobs
121355b4669Sjacobs /* allows a NULL collection that is "empty" or "no value" */
122355b4669Sjacobs if (collection != NULL) {
123355b4669Sjacobs papi_status_t status = PAPI_OK;
124355b4669Sjacobs int i;
125355b4669Sjacobs
126355b4669Sjacobs for (i = 0; ((collection[i] != NULL) && (status == PAPI_OK));
127355b4669Sjacobs i++) {
128355b4669Sjacobs papi_attribute_t *a = collection[i];
129355b4669Sjacobs
130355b4669Sjacobs status = papiAttributeListAddValue(&result,
1313cf7d3e9SToomas Soome PAPI_ATTR_APPEND, a->name, a->type, NULL);
132355b4669Sjacobs if ((status == PAPI_OK) && (a->values != NULL)) {
133355b4669Sjacobs int j;
134355b4669Sjacobs
135355b4669Sjacobs for (j = 0; ((a->values[j] != NULL) &&
136355b4669Sjacobs (status == PAPI_OK)); j++)
137355b4669Sjacobs status = papiAttributeListAddValue(
1383cf7d3e9SToomas Soome &result, PAPI_ATTR_APPEND,
1393cf7d3e9SToomas Soome a->name, a->type, a->values[j]);
140355b4669Sjacobs }
141355b4669Sjacobs }
142355b4669Sjacobs if (status != PAPI_OK) {
143355b4669Sjacobs papiAttributeListFree(result);
144355b4669Sjacobs result = NULL;
145355b4669Sjacobs }
146355b4669Sjacobs }
147355b4669Sjacobs
148355b4669Sjacobs return (result);
149355b4669Sjacobs }
150355b4669Sjacobs
151355b4669Sjacobs static papi_attribute_value_t *
papiAttributeValueDup(papi_attribute_value_type_t type,papi_attribute_value_t * v)152355b4669Sjacobs papiAttributeValueDup(papi_attribute_value_type_t type,
153355b4669Sjacobs papi_attribute_value_t *v)
154355b4669Sjacobs {
155355b4669Sjacobs papi_attribute_value_t *result = NULL;
156355b4669Sjacobs
157355b4669Sjacobs if ((v != NULL) && ((result = calloc(1, sizeof (*result))) != NULL)) {
158355b4669Sjacobs switch (type) {
159355b4669Sjacobs case PAPI_STRING:
160355b4669Sjacobs if (v->string == NULL) {
161355b4669Sjacobs free(result);
162355b4669Sjacobs result = NULL;
163*f4593de7SToomas Soome } else {
164355b4669Sjacobs result->string = strdup(v->string);
165*f4593de7SToomas Soome }
166355b4669Sjacobs break;
167355b4669Sjacobs case PAPI_INTEGER:
168355b4669Sjacobs result->integer = v->integer;
169355b4669Sjacobs break;
170355b4669Sjacobs case PAPI_BOOLEAN:
171355b4669Sjacobs result->boolean = v->boolean;
172355b4669Sjacobs break;
173355b4669Sjacobs case PAPI_RANGE:
174355b4669Sjacobs result->range.lower = v->range.lower;
175355b4669Sjacobs result->range.upper = v->range.upper;
176355b4669Sjacobs break;
177355b4669Sjacobs case PAPI_RESOLUTION:
178355b4669Sjacobs result->resolution.xres = v->resolution.xres;
179355b4669Sjacobs result->resolution.yres = v->resolution.yres;
180355b4669Sjacobs result->resolution.units = v->resolution.units;
181355b4669Sjacobs break;
182355b4669Sjacobs case PAPI_DATETIME:
183355b4669Sjacobs result->datetime = v->datetime;
184355b4669Sjacobs break;
185355b4669Sjacobs case PAPI_COLLECTION:
186355b4669Sjacobs result->collection = collection_dup(v->collection);
187355b4669Sjacobs break;
188355b4669Sjacobs case PAPI_METADATA:
189355b4669Sjacobs result->metadata = v->metadata;
190355b4669Sjacobs break;
191355b4669Sjacobs default: /* unknown type, fail to duplicate */
192355b4669Sjacobs free(result);
193355b4669Sjacobs result = NULL;
194355b4669Sjacobs }
195355b4669Sjacobs }
196355b4669Sjacobs
197355b4669Sjacobs return (result);
198355b4669Sjacobs }
199355b4669Sjacobs
200355b4669Sjacobs static papi_attribute_t *
papiAttributeAlloc(char * name,papi_attribute_value_type_t type)201355b4669Sjacobs papiAttributeAlloc(char *name, papi_attribute_value_type_t type)
202355b4669Sjacobs {
203355b4669Sjacobs papi_attribute_t *result = NULL;
204355b4669Sjacobs
205355b4669Sjacobs if ((result = calloc(1, sizeof (*result))) != NULL) {
206355b4669Sjacobs result->name = strdup(name);
207355b4669Sjacobs result->type = type;
208355b4669Sjacobs }
209355b4669Sjacobs
210355b4669Sjacobs return (result);
211355b4669Sjacobs }
212355b4669Sjacobs
213355b4669Sjacobs static papi_status_t
papiAttributeListAppendValue(papi_attribute_value_t *** values,papi_attribute_value_type_t type,papi_attribute_value_t * value)214355b4669Sjacobs papiAttributeListAppendValue(papi_attribute_value_t ***values,
215355b4669Sjacobs papi_attribute_value_type_t type,
216355b4669Sjacobs papi_attribute_value_t *value)
217355b4669Sjacobs {
218355b4669Sjacobs
219355b4669Sjacobs if (values == NULL)
220355b4669Sjacobs return (PAPI_BAD_ARGUMENT);
221355b4669Sjacobs
222355b4669Sjacobs if (value != NULL) { /* this allows "empty" attributes */
223355b4669Sjacobs papi_attribute_value_t *tmp = NULL;
224355b4669Sjacobs
225355b4669Sjacobs if ((tmp = papiAttributeValueDup(type, value)) == NULL)
226355b4669Sjacobs return (PAPI_TEMPORARY_ERROR);
227355b4669Sjacobs
228355b4669Sjacobs list_append(values, tmp);
229355b4669Sjacobs }
230355b4669Sjacobs
231355b4669Sjacobs return (PAPI_OK);
232355b4669Sjacobs }
233355b4669Sjacobs
234355b4669Sjacobs papi_status_t
papiAttributeListAddValue(papi_attribute_t *** list,int flgs,char * name,papi_attribute_value_type_t type,papi_attribute_value_t * value)235355b4669Sjacobs papiAttributeListAddValue(papi_attribute_t ***list, int flgs,
236355b4669Sjacobs char *name, papi_attribute_value_type_t type,
237355b4669Sjacobs papi_attribute_value_t *value)
238355b4669Sjacobs {
239355b4669Sjacobs papi_status_t result;
240355b4669Sjacobs int flags = flgs;
241355b4669Sjacobs papi_attribute_t *attribute = NULL;
242355b4669Sjacobs papi_attribute_value_t **values = NULL;
243355b4669Sjacobs
244355b4669Sjacobs if ((list == NULL) || (name == NULL))
245355b4669Sjacobs return (PAPI_BAD_ARGUMENT);
246355b4669Sjacobs
247355b4669Sjacobs if ((type == PAPI_RANGE) && (value != NULL) &&
248355b4669Sjacobs (value->range.lower > value->range.upper))
249355b4669Sjacobs return (PAPI_BAD_ARGUMENT); /* RANGE must have min <= max */
250355b4669Sjacobs
251355b4669Sjacobs if (flags == 0) /* if it wasn't set, set a default behaviour */
252355b4669Sjacobs flags = PAPI_ATTR_APPEND;
253355b4669Sjacobs
254355b4669Sjacobs /* look for an existing one */
255355b4669Sjacobs attribute = papiAttributeListFind(*list, name);
256355b4669Sjacobs
257355b4669Sjacobs if (((flags & PAPI_ATTR_EXCL) != 0) && (attribute != NULL))
258355b4669Sjacobs return (PAPI_CONFLICT); /* EXISTS */
259355b4669Sjacobs
260355b4669Sjacobs if (((flags & PAPI_ATTR_REPLACE) == 0) && (attribute != NULL) &&
261355b4669Sjacobs (attribute->type != type))
262355b4669Sjacobs return (PAPI_CONFLICT); /* TYPE CONFLICT */
263355b4669Sjacobs
264355b4669Sjacobs /* if we don't have one, create it and add it to the list */
265355b4669Sjacobs if ((attribute == NULL) &&
266355b4669Sjacobs ((attribute = papiAttributeAlloc(name, type)) != NULL))
267355b4669Sjacobs list_append(list, attribute);
268355b4669Sjacobs
269355b4669Sjacobs /* if we don't have one by now, it's most likely an alloc fail */
270355b4669Sjacobs if (attribute == NULL)
271355b4669Sjacobs return (PAPI_TEMPORARY_ERROR);
272355b4669Sjacobs
273355b4669Sjacobs /*
274355b4669Sjacobs * if we are replacing, clear any existing values, but don't free
275355b4669Sjacobs * until after we have replaced the values, in case we are replacing
276355b4669Sjacobs * a collection with a relocated version of the original collection.
277355b4669Sjacobs */
278355b4669Sjacobs if (((flags & PAPI_ATTR_REPLACE) != 0) && (attribute->values != NULL)) {
279355b4669Sjacobs values = attribute->values;
280355b4669Sjacobs attribute->values = NULL;
281355b4669Sjacobs }
282355b4669Sjacobs
283355b4669Sjacobs attribute->type = type;
284355b4669Sjacobs
285355b4669Sjacobs result = papiAttributeListAppendValue(&attribute->values, type, value);
286355b4669Sjacobs
287355b4669Sjacobs /* free old values if we replaced them */
288355b4669Sjacobs if (values != NULL)
289355b4669Sjacobs papiAttributeValuesFree(type, values);
290355b4669Sjacobs
291355b4669Sjacobs return (result);
292355b4669Sjacobs }
293355b4669Sjacobs
294355b4669Sjacobs papi_status_t
papiAttributeListAddString(papi_attribute_t *** list,int flags,char * name,char * string)295355b4669Sjacobs papiAttributeListAddString(papi_attribute_t ***list, int flags,
296355b4669Sjacobs char *name, char *string)
297355b4669Sjacobs {
298355b4669Sjacobs papi_attribute_value_t v;
299355b4669Sjacobs
300355b4669Sjacobs v.string = (char *)string;
301355b4669Sjacobs return (papiAttributeListAddValue(list, flags, name, PAPI_STRING, &v));
302355b4669Sjacobs }
303355b4669Sjacobs
304355b4669Sjacobs papi_status_t
papiAttributeListAddInteger(papi_attribute_t *** list,int flags,char * name,int integer)305355b4669Sjacobs papiAttributeListAddInteger(papi_attribute_t ***list, int flags,
306355b4669Sjacobs char *name, int integer)
307355b4669Sjacobs {
308355b4669Sjacobs papi_attribute_value_t v;
309355b4669Sjacobs
310355b4669Sjacobs v.integer = integer;
311355b4669Sjacobs return (papiAttributeListAddValue(list, flags, name, PAPI_INTEGER, &v));
312355b4669Sjacobs }
313355b4669Sjacobs
314355b4669Sjacobs papi_status_t
papiAttributeListAddBoolean(papi_attribute_t *** list,int flags,char * name,char boolean)315355b4669Sjacobs papiAttributeListAddBoolean(papi_attribute_t ***list, int flags,
316355b4669Sjacobs char *name, char boolean)
317355b4669Sjacobs {
318355b4669Sjacobs papi_attribute_value_t v;
319355b4669Sjacobs
320355b4669Sjacobs v.boolean = boolean;
321355b4669Sjacobs return (papiAttributeListAddValue(list, flags, name, PAPI_BOOLEAN, &v));
322355b4669Sjacobs }
323355b4669Sjacobs
324355b4669Sjacobs papi_status_t
papiAttributeListAddRange(papi_attribute_t *** list,int flags,char * name,int lower,int upper)325355b4669Sjacobs papiAttributeListAddRange(papi_attribute_t ***list, int flags,
326355b4669Sjacobs char *name, int lower, int upper)
327355b4669Sjacobs {
328355b4669Sjacobs papi_attribute_value_t v;
329355b4669Sjacobs
330355b4669Sjacobs v.range.lower = lower;
331355b4669Sjacobs v.range.upper = upper;
332355b4669Sjacobs return (papiAttributeListAddValue(list, flags, name, PAPI_RANGE, &v));
333355b4669Sjacobs }
334355b4669Sjacobs
335355b4669Sjacobs papi_status_t
papiAttributeListAddResolution(papi_attribute_t *** list,int flags,char * name,int xres,int yres,papi_resolution_unit_t units)336355b4669Sjacobs papiAttributeListAddResolution(papi_attribute_t ***list, int flags,
3373cf7d3e9SToomas Soome char *name, int xres, int yres, papi_resolution_unit_t units)
338355b4669Sjacobs {
339355b4669Sjacobs papi_attribute_value_t v;
340355b4669Sjacobs
341355b4669Sjacobs v.resolution.xres = xres;
342355b4669Sjacobs v.resolution.yres = yres;
343355b4669Sjacobs v.resolution.units = units;
344355b4669Sjacobs return (papiAttributeListAddValue(list, flags, name,
345355b4669Sjacobs PAPI_RESOLUTION, &v));
346355b4669Sjacobs }
347355b4669Sjacobs
348355b4669Sjacobs papi_status_t
papiAttributeListAddDatetime(papi_attribute_t *** list,int flags,char * name,time_t datetime)349355b4669Sjacobs papiAttributeListAddDatetime(papi_attribute_t ***list, int flags,
350355b4669Sjacobs char *name, time_t datetime)
351355b4669Sjacobs {
352355b4669Sjacobs papi_attribute_value_t v;
353355b4669Sjacobs
354355b4669Sjacobs v.datetime = datetime;
355355b4669Sjacobs return (papiAttributeListAddValue(list, flags, name,
356355b4669Sjacobs PAPI_DATETIME, &v));
357355b4669Sjacobs }
358355b4669Sjacobs
359355b4669Sjacobs papi_status_t
papiAttributeListAddCollection(papi_attribute_t *** list,int flags,char * name,papi_attribute_t ** collection)360355b4669Sjacobs papiAttributeListAddCollection(papi_attribute_t ***list, int flags,
361355b4669Sjacobs char *name, papi_attribute_t **collection)
362355b4669Sjacobs {
363355b4669Sjacobs papi_attribute_value_t v;
364355b4669Sjacobs
365355b4669Sjacobs v.collection = (papi_attribute_t **)collection;
366355b4669Sjacobs return (papiAttributeListAddValue(list, flags, name,
367355b4669Sjacobs PAPI_COLLECTION, &v));
368355b4669Sjacobs }
369355b4669Sjacobs
370355b4669Sjacobs papi_status_t
papiAttributeListAddMetadata(papi_attribute_t *** list,int flags,char * name,papi_metadata_t metadata)371355b4669Sjacobs papiAttributeListAddMetadata(papi_attribute_t ***list, int flags,
372355b4669Sjacobs char *name, papi_metadata_t metadata)
373355b4669Sjacobs {
374355b4669Sjacobs papi_attribute_value_t v;
375355b4669Sjacobs
376355b4669Sjacobs v.metadata = metadata;
377355b4669Sjacobs return (papiAttributeListAddValue(list, flags, name,
378355b4669Sjacobs PAPI_METADATA, &v));
379355b4669Sjacobs }
380355b4669Sjacobs
381355b4669Sjacobs papi_status_t
papiAttributeListDelete(papi_attribute_t *** list,char * name)382355b4669Sjacobs papiAttributeListDelete(papi_attribute_t ***list, char *name)
383355b4669Sjacobs {
384355b4669Sjacobs papi_attribute_t *attribute;
385355b4669Sjacobs
386355b4669Sjacobs if ((list == NULL) || (name == NULL))
387355b4669Sjacobs return (PAPI_BAD_ARGUMENT);
388355b4669Sjacobs
389355b4669Sjacobs if ((attribute = papiAttributeListFind(*list, name)) == NULL)
390355b4669Sjacobs return (PAPI_NOT_FOUND);
391355b4669Sjacobs
3920a44ef6dSjacobs list_remove(list, attribute);
393355b4669Sjacobs papiAttributeFree(attribute);
394355b4669Sjacobs
395355b4669Sjacobs return (PAPI_OK);
396355b4669Sjacobs }
397355b4669Sjacobs
398355b4669Sjacobs papi_attribute_t *
papiAttributeListFind(papi_attribute_t ** list,char * name)399355b4669Sjacobs papiAttributeListFind(papi_attribute_t **list, char *name)
400355b4669Sjacobs {
401355b4669Sjacobs int i;
402355b4669Sjacobs if ((list == NULL) || (name == NULL))
403355b4669Sjacobs return (NULL);
404355b4669Sjacobs
405355b4669Sjacobs for (i = 0; list[i] != NULL; i++)
406355b4669Sjacobs if (strcasecmp(list[i]->name, name) == 0)
407355b4669Sjacobs return ((papi_attribute_t *)list[i]);
408355b4669Sjacobs
409355b4669Sjacobs return (NULL);
410355b4669Sjacobs }
411355b4669Sjacobs
412355b4669Sjacobs papi_attribute_t *
papiAttributeListGetNext(papi_attribute_t ** list,void ** iter)413355b4669Sjacobs papiAttributeListGetNext(papi_attribute_t **list, void **iter)
414355b4669Sjacobs {
415355b4669Sjacobs papi_attribute_t **tmp, *result;
416355b4669Sjacobs
417355b4669Sjacobs if ((list == NULL) && (iter == NULL))
418355b4669Sjacobs return (NULL);
419355b4669Sjacobs
420355b4669Sjacobs if (*iter == NULL)
421355b4669Sjacobs *iter = list;
422355b4669Sjacobs
423355b4669Sjacobs tmp = *iter;
424355b4669Sjacobs result = *tmp;
425355b4669Sjacobs *iter = ++tmp;
426355b4669Sjacobs
427355b4669Sjacobs return (result);
428355b4669Sjacobs }
429355b4669Sjacobs
430355b4669Sjacobs papi_status_t
papiAttributeListGetValue(papi_attribute_t ** list,void ** iter,char * name,papi_attribute_value_type_t type,papi_attribute_value_t ** value)431355b4669Sjacobs papiAttributeListGetValue(papi_attribute_t **list, void **iter,
432355b4669Sjacobs char *name, papi_attribute_value_type_t type,
433355b4669Sjacobs papi_attribute_value_t **value)
434355b4669Sjacobs {
435355b4669Sjacobs papi_attribute_value_t **tmp;
436355b4669Sjacobs void *fodder = NULL;
437355b4669Sjacobs
438355b4669Sjacobs if ((list == NULL) || ((name == NULL) && (iter == NULL)) ||
439355b4669Sjacobs (value == NULL))
440355b4669Sjacobs return (PAPI_BAD_ARGUMENT);
441355b4669Sjacobs
442355b4669Sjacobs if (iter == NULL)
443355b4669Sjacobs iter = &fodder;
444355b4669Sjacobs
445355b4669Sjacobs if ((iter == NULL) || (*iter == NULL)) {
446355b4669Sjacobs papi_attribute_t *attr = papiAttributeListFind(list, name);
447355b4669Sjacobs
448355b4669Sjacobs if (attr == NULL)
449355b4669Sjacobs return (PAPI_NOT_FOUND);
450355b4669Sjacobs
451355b4669Sjacobs if (attr->type != type)
452355b4669Sjacobs return (PAPI_NOT_POSSIBLE);
453355b4669Sjacobs
454355b4669Sjacobs tmp = attr->values;
455*f4593de7SToomas Soome } else {
456355b4669Sjacobs tmp = *iter;
457*f4593de7SToomas Soome }
458355b4669Sjacobs
459355b4669Sjacobs if (tmp == NULL)
460355b4669Sjacobs return (PAPI_NOT_FOUND);
461355b4669Sjacobs
462355b4669Sjacobs *value = *tmp;
463355b4669Sjacobs *iter = ++tmp;
464355b4669Sjacobs
465355b4669Sjacobs if (*value == NULL)
466355b4669Sjacobs return (PAPI_GONE);
467355b4669Sjacobs
468355b4669Sjacobs return (PAPI_OK);
469355b4669Sjacobs }
470355b4669Sjacobs
471355b4669Sjacobs papi_status_t
papiAttributeListGetString(papi_attribute_t ** list,void ** iter,char * name,char ** vptr)472355b4669Sjacobs papiAttributeListGetString(papi_attribute_t **list, void **iter,
473355b4669Sjacobs char *name, char **vptr)
474355b4669Sjacobs {
475355b4669Sjacobs papi_status_t status;
476355b4669Sjacobs papi_attribute_value_t *value = NULL;
477355b4669Sjacobs
478355b4669Sjacobs if (vptr == NULL)
479355b4669Sjacobs return (PAPI_BAD_ARGUMENT);
480355b4669Sjacobs
481355b4669Sjacobs status = papiAttributeListGetValue(list, iter, name,
482355b4669Sjacobs PAPI_STRING, &value);
483355b4669Sjacobs if (status == PAPI_OK)
484355b4669Sjacobs *vptr = value->string;
485355b4669Sjacobs
486355b4669Sjacobs return (status);
487355b4669Sjacobs }
488355b4669Sjacobs
489355b4669Sjacobs papi_status_t
papiAttributeListGetInteger(papi_attribute_t ** list,void ** iter,char * name,int * vptr)490355b4669Sjacobs papiAttributeListGetInteger(papi_attribute_t **list, void **iter,
491355b4669Sjacobs char *name, int *vptr)
492355b4669Sjacobs {
493355b4669Sjacobs papi_status_t status;
494355b4669Sjacobs papi_attribute_value_t *value = NULL;
495355b4669Sjacobs
496355b4669Sjacobs if (vptr == NULL)
497355b4669Sjacobs return (PAPI_BAD_ARGUMENT);
498355b4669Sjacobs
499355b4669Sjacobs status = papiAttributeListGetValue(list, iter, name,
500355b4669Sjacobs PAPI_INTEGER, &value);
501355b4669Sjacobs if (status == PAPI_OK)
502355b4669Sjacobs *vptr = value->integer;
503355b4669Sjacobs
504355b4669Sjacobs return (status);
505355b4669Sjacobs }
506355b4669Sjacobs
507355b4669Sjacobs papi_status_t
papiAttributeListGetBoolean(papi_attribute_t ** list,void ** iter,char * name,char * vptr)508355b4669Sjacobs papiAttributeListGetBoolean(papi_attribute_t **list, void **iter,
509355b4669Sjacobs char *name, char *vptr)
510355b4669Sjacobs {
511355b4669Sjacobs papi_status_t status;
512355b4669Sjacobs papi_attribute_value_t *value = NULL;
513355b4669Sjacobs
514355b4669Sjacobs if (vptr == NULL)
515355b4669Sjacobs return (PAPI_BAD_ARGUMENT);
516355b4669Sjacobs
517355b4669Sjacobs status = papiAttributeListGetValue(list, iter, name,
518355b4669Sjacobs PAPI_BOOLEAN, &value);
519355b4669Sjacobs if (status == PAPI_OK)
520355b4669Sjacobs *vptr = value->boolean;
521355b4669Sjacobs
522355b4669Sjacobs return (status);
523355b4669Sjacobs }
524355b4669Sjacobs
525355b4669Sjacobs papi_status_t
papiAttributeListGetRange(papi_attribute_t ** list,void ** iter,char * name,int * min,int * max)526355b4669Sjacobs papiAttributeListGetRange(papi_attribute_t **list, void **iter,
527355b4669Sjacobs char *name, int *min, int *max)
528355b4669Sjacobs {
529355b4669Sjacobs papi_status_t status;
530355b4669Sjacobs papi_attribute_value_t *value = NULL;
531355b4669Sjacobs
532355b4669Sjacobs if ((min == NULL) || (max == NULL))
533355b4669Sjacobs return (PAPI_BAD_ARGUMENT);
534355b4669Sjacobs
535355b4669Sjacobs status = papiAttributeListGetValue(list, iter, name,
536355b4669Sjacobs PAPI_RANGE, &value);
537355b4669Sjacobs if (status == PAPI_OK) {
538355b4669Sjacobs *min = value->range.lower;
539355b4669Sjacobs *max = value->range.upper;
540355b4669Sjacobs }
541355b4669Sjacobs
542355b4669Sjacobs return (status);
543355b4669Sjacobs }
544355b4669Sjacobs
545355b4669Sjacobs papi_status_t
papiAttributeListGetResolution(papi_attribute_t ** list,void ** iter,char * name,int * x,int * y,papi_resolution_unit_t * units)546355b4669Sjacobs papiAttributeListGetResolution(papi_attribute_t **list, void **iter,
5473cf7d3e9SToomas Soome char *name, int *x, int *y, papi_resolution_unit_t *units)
548355b4669Sjacobs {
549355b4669Sjacobs papi_status_t status;
550355b4669Sjacobs papi_attribute_value_t *value = NULL;
551355b4669Sjacobs
552355b4669Sjacobs if ((x == NULL) || (y == NULL) || (units == NULL))
553355b4669Sjacobs return (PAPI_BAD_ARGUMENT);
554355b4669Sjacobs
555355b4669Sjacobs status = papiAttributeListGetValue(list, iter, name,
556355b4669Sjacobs PAPI_RESOLUTION, &value);
557355b4669Sjacobs if (status == PAPI_OK) {
558355b4669Sjacobs *x = value->resolution.xres;
559355b4669Sjacobs *y = value->resolution.yres;
560355b4669Sjacobs *units = value->resolution.units;
561355b4669Sjacobs }
562355b4669Sjacobs
563355b4669Sjacobs return (status);
564355b4669Sjacobs }
565355b4669Sjacobs
566355b4669Sjacobs papi_status_t
papiAttributeListGetDatetime(papi_attribute_t ** list,void ** iter,char * name,time_t * dt)567355b4669Sjacobs papiAttributeListGetDatetime(papi_attribute_t **list, void **iter,
568355b4669Sjacobs char *name, time_t *dt)
569355b4669Sjacobs {
570355b4669Sjacobs papi_status_t status;
571355b4669Sjacobs papi_attribute_value_t *value = NULL;
572355b4669Sjacobs
573355b4669Sjacobs if (dt == NULL)
574355b4669Sjacobs return (PAPI_BAD_ARGUMENT);
575355b4669Sjacobs
576355b4669Sjacobs status = papiAttributeListGetValue(list, iter, name,
577355b4669Sjacobs PAPI_DATETIME, &value);
578355b4669Sjacobs if (status == PAPI_OK) {
579355b4669Sjacobs *dt = value->datetime;
580355b4669Sjacobs }
581355b4669Sjacobs
582355b4669Sjacobs return (status);
583355b4669Sjacobs }
584355b4669Sjacobs
585355b4669Sjacobs papi_status_t
papiAttributeListGetCollection(papi_attribute_t ** list,void ** iter,char * name,papi_attribute_t *** collection)586355b4669Sjacobs papiAttributeListGetCollection(papi_attribute_t **list, void **iter,
587355b4669Sjacobs char *name, papi_attribute_t ***collection)
588355b4669Sjacobs {
589355b4669Sjacobs papi_status_t status;
590355b4669Sjacobs papi_attribute_value_t *value = NULL;
591355b4669Sjacobs
592355b4669Sjacobs if (collection == NULL)
593355b4669Sjacobs return (PAPI_BAD_ARGUMENT);
594355b4669Sjacobs
595355b4669Sjacobs status = papiAttributeListGetValue(list, iter, name,
596355b4669Sjacobs PAPI_COLLECTION, &value);
597355b4669Sjacobs if (status == PAPI_OK) {
598355b4669Sjacobs *collection = value->collection;
599355b4669Sjacobs }
600355b4669Sjacobs
601355b4669Sjacobs return (status);
602355b4669Sjacobs }
603355b4669Sjacobs
604355b4669Sjacobs papi_status_t
papiAttributeListGetMetadata(papi_attribute_t ** list,void ** iter,char * name,papi_metadata_t * vptr)605355b4669Sjacobs papiAttributeListGetMetadata(papi_attribute_t **list, void **iter,
606355b4669Sjacobs char *name, papi_metadata_t *vptr)
607355b4669Sjacobs {
608355b4669Sjacobs papi_status_t status;
609355b4669Sjacobs papi_attribute_value_t *value = NULL;
610355b4669Sjacobs
611355b4669Sjacobs if (vptr == NULL)
612355b4669Sjacobs return (PAPI_BAD_ARGUMENT);
613355b4669Sjacobs
614355b4669Sjacobs status = papiAttributeListGetValue(list, iter, name,
615355b4669Sjacobs PAPI_METADATA, &value);
616355b4669Sjacobs if (status == PAPI_OK)
617355b4669Sjacobs *vptr = value->metadata;
618355b4669Sjacobs
619355b4669Sjacobs return (status);
620355b4669Sjacobs }
621355b4669Sjacobs
622355b4669Sjacobs
62343b9c050Sjacobs /* The string is modified by this call */
624355b4669Sjacobs static char *
regvalue(regmatch_t match,char * string)62543b9c050Sjacobs regvalue(regmatch_t match, char *string)
626355b4669Sjacobs {
627355b4669Sjacobs char *result = NULL;
62843b9c050Sjacobs if (match.rm_so != match.rm_eo) {
62943b9c050Sjacobs result = string + match.rm_so;
63043b9c050Sjacobs *(result + (match.rm_eo - match.rm_so)) = '\0';
631355b4669Sjacobs }
632355b4669Sjacobs return (result);
63343b9c050Sjacobs }
634355b4669Sjacobs
63543b9c050Sjacobs static papi_attribute_value_type_t
_process_value(char * string,char *** parts)63643b9c050Sjacobs _process_value(char *string, char ***parts)
63743b9c050Sjacobs {
63843b9c050Sjacobs int i;
63943b9c050Sjacobs static struct {
64043b9c050Sjacobs papi_attribute_value_type_t type;
64143b9c050Sjacobs size_t vals;
64243b9c050Sjacobs char *expression;
64343b9c050Sjacobs int compiled;
64443b9c050Sjacobs regex_t re;
64543b9c050Sjacobs } types[] = {
64643b9c050Sjacobs { PAPI_BOOLEAN, 1, "^(true|false|yes|no)$", 0 },
64743b9c050Sjacobs { PAPI_COLLECTION, 1, "^\\{(.+)\\}$", 0 },
64843b9c050Sjacobs /* PAPI_DATETIME is unsupported, too much like an integer */
64943b9c050Sjacobs { PAPI_INTEGER, 1, "^([+-]{0,1}[[:digit:]]+)$", 0 },
650fa9d6f9fSsg223391 { PAPI_RANGE, 3, "^([[:digit:]]*)-([[:digit:]]*)$", 0 },
65143b9c050Sjacobs { PAPI_RESOLUTION, 4, "^([[:digit:]]+)x([[:digit:]]+)dp(i|c)$",
65243b9c050Sjacobs 0 },
653*f4593de7SToomas Soome 0
65443b9c050Sjacobs };
65543b9c050Sjacobs regmatch_t matches[4];
656355b4669Sjacobs
65743b9c050Sjacobs for (i = 0; i < 5; i++) {
65843b9c050Sjacobs int j;
65943b9c050Sjacobs
66043b9c050Sjacobs if (types[i].compiled == 0) {
66143b9c050Sjacobs (void) regcomp(&(types[i].re), types[i].expression,
66243b9c050Sjacobs REG_EXTENDED|REG_ICASE);
66343b9c050Sjacobs types[i].compiled = 1;
66443b9c050Sjacobs }
66543b9c050Sjacobs if (regexec(&(types[i].re), string, (size_t)types[i].vals,
66643b9c050Sjacobs matches, 0) == REG_NOMATCH)
66743b9c050Sjacobs continue;
66843b9c050Sjacobs
66943b9c050Sjacobs for (j = 0; j < types[i].vals; j++)
67043b9c050Sjacobs list_append(parts, regvalue(matches[j], string));
67143b9c050Sjacobs return (types[i].type);
67243b9c050Sjacobs }
67343b9c050Sjacobs
67443b9c050Sjacobs list_append(parts, string);
67543b9c050Sjacobs return (PAPI_STRING);
67643b9c050Sjacobs }
67743b9c050Sjacobs
67843b9c050Sjacobs static void
_add_attribute_value(papi_attribute_value_t *** list,papi_attribute_value_type_t type,papi_attribute_value_type_t dtype,char ** parts)67943b9c050Sjacobs _add_attribute_value(papi_attribute_value_t ***list,
68043b9c050Sjacobs papi_attribute_value_type_t type,
68143b9c050Sjacobs papi_attribute_value_type_t dtype, char **parts)
68243b9c050Sjacobs {
68343b9c050Sjacobs papi_attribute_value_t *value = calloc(1, sizeof (*value));
68443b9c050Sjacobs
68543b9c050Sjacobs switch (type) {
68643b9c050Sjacobs case PAPI_STRING:
68743b9c050Sjacobs value->string = strdup(parts[0]);
68843b9c050Sjacobs list_append(list, value);
68943b9c050Sjacobs break;
69043b9c050Sjacobs case PAPI_BOOLEAN:
69143b9c050Sjacobs value->boolean = PAPI_TRUE;
69243b9c050Sjacobs if ((strcasecmp(parts[0], "false") == 0) ||
69343b9c050Sjacobs (strcasecmp(parts[0], "no") == 0))
69443b9c050Sjacobs value->boolean = PAPI_FALSE;
69543b9c050Sjacobs list_append(list, value);
69643b9c050Sjacobs break;
69743b9c050Sjacobs case PAPI_INTEGER:
69843b9c050Sjacobs value->integer = atoi(parts[0]);
69943b9c050Sjacobs list_append(list, value);
70043b9c050Sjacobs break;
70143b9c050Sjacobs case PAPI_RANGE:
702fa9d6f9fSsg223391 if (dtype == PAPI_INTEGER) {
703fa9d6f9fSsg223391 if (atoi(parts[0]) < 0) {
704fa9d6f9fSsg223391 /*
705fa9d6f9fSsg223391 * Handles -P -x case
706fa9d6f9fSsg223391 * which prints from page number 1
707fa9d6f9fSsg223391 * till page number x
708fa9d6f9fSsg223391 */
709fa9d6f9fSsg223391 value->range.lower = 1;
710fa9d6f9fSsg223391 value->range.upper = 0 - (atoi(parts[0]));
711fa9d6f9fSsg223391 } else {
71243b9c050Sjacobs value->range.lower = value->range.upper
71343b9c050Sjacobs = atoi(parts[0]);
714fa9d6f9fSsg223391 }
715fa9d6f9fSsg223391 } else if (dtype == PAPI_RANGE) {
716fa9d6f9fSsg223391 if (parts[2] == NULL) {
717fa9d6f9fSsg223391 value->range.lower = atoi(parts[1]);
718fa9d6f9fSsg223391 /*
719fa9d6f9fSsg223391 * Imposing an artificial limit on
720fa9d6f9fSsg223391 * the upper bound for page range.
721fa9d6f9fSsg223391 */
722fa9d6f9fSsg223391 value->range.upper = MAX_PAGES;
723fa9d6f9fSsg223391 } else if ((parts[1] != NULL) && (parts[2] != NULL)) {
72443b9c050Sjacobs value->range.lower = atoi(parts[1]);
72543b9c050Sjacobs value->range.upper = atoi(parts[2]);
72643b9c050Sjacobs }
727fa9d6f9fSsg223391 }
72843b9c050Sjacobs list_append(list, value);
72943b9c050Sjacobs break;
73043b9c050Sjacobs case PAPI_RESOLUTION:
73143b9c050Sjacobs value->resolution.xres = atoi(parts[1]);
73243b9c050Sjacobs value->resolution.yres = atoi(parts[2]);
73343b9c050Sjacobs if (parts[3][0] == 'i')
73443b9c050Sjacobs value->resolution.units = PAPI_RES_PER_INCH;
73543b9c050Sjacobs else
73643b9c050Sjacobs value->resolution.units = PAPI_RES_PER_CM;
73743b9c050Sjacobs list_append(list, value);
73843b9c050Sjacobs break;
73943b9c050Sjacobs case PAPI_COLLECTION:
74043b9c050Sjacobs papiAttributeListFromString(&(value->collection), 0, parts[0]);
74143b9c050Sjacobs list_append(list, value);
74243b9c050Sjacobs break;
74343b9c050Sjacobs }
74443b9c050Sjacobs }
745355b4669Sjacobs
746355b4669Sjacobs static papi_status_t
_papiAttributeFromStrings(papi_attribute_t *** list,int flags,char * key,char ** values)74743b9c050Sjacobs _papiAttributeFromStrings(papi_attribute_t ***list, int flags,
74843b9c050Sjacobs char *key, char **values)
749355b4669Sjacobs {
75043b9c050Sjacobs int i;
751355b4669Sjacobs papi_status_t result = PAPI_OK;
75243b9c050Sjacobs papi_attribute_t *attr = calloc(1, sizeof (*attr));
753355b4669Sjacobs
75443b9c050Sjacobs /* these are specified in the papi spec as ranges */
75543b9c050Sjacobs char *ranges[] = { "copies-supported", "job-impressions-supported",
75643b9c050Sjacobs "job-k-octets-supported",
75743b9c050Sjacobs "job-media-sheets-supported", "page-ranges",
75843b9c050Sjacobs NULL };
759355b4669Sjacobs
76043b9c050Sjacobs if ((attr == NULL) || ((attr->name = strdup(key)) == NULL))
76143b9c050Sjacobs return (PAPI_TEMPORARY_ERROR);
762355b4669Sjacobs
76343b9c050Sjacobs attr->type = PAPI_METADATA;
76443b9c050Sjacobs /* these are known ranges */
76543b9c050Sjacobs for (i = 0; ranges[i] != NULL; i++)
76643b9c050Sjacobs if (strcasecmp(attr->name, ranges[i]) == 0) {
767355b4669Sjacobs attr->type = PAPI_RANGE;
76843b9c050Sjacobs break;
769355b4669Sjacobs }
770355b4669Sjacobs
77143b9c050Sjacobs if (values != NULL) {
77243b9c050Sjacobs papi_attribute_value_t **vals = NULL;
77343b9c050Sjacobs
77443b9c050Sjacobs for (i = 0; values[i] != NULL; i++) {
77543b9c050Sjacobs papi_attribute_value_type_t dtype;
77643b9c050Sjacobs char **parts = NULL;
77743b9c050Sjacobs
77843b9c050Sjacobs dtype = _process_value(values[i], &parts);
77943b9c050Sjacobs if (attr->type == PAPI_METADATA)
78043b9c050Sjacobs attr->type = dtype;
78143b9c050Sjacobs _add_attribute_value(&vals, attr->type, dtype, parts);
78243b9c050Sjacobs free(parts);
78343b9c050Sjacobs }
78443b9c050Sjacobs attr->values = vals;
785355b4669Sjacobs }
786355b4669Sjacobs
78743b9c050Sjacobs list_append(list, attr);
788355b4669Sjacobs
789355b4669Sjacobs return (result);
79043b9c050Sjacobs }
791355b4669Sjacobs
792355b4669Sjacobs static papi_status_t
_parse_attribute_list(papi_attribute_t *** list,int flags,char * string)79343b9c050Sjacobs _parse_attribute_list(papi_attribute_t ***list, int flags, char *string)
794355b4669Sjacobs {
795355b4669Sjacobs papi_status_t result = PAPI_OK;
79643b9c050Sjacobs char *ptr;
797355b4669Sjacobs
79843b9c050Sjacobs if ((list == NULL) || (string == NULL))
79943b9c050Sjacobs return (PAPI_BAD_ARGUMENT);
800355b4669Sjacobs
80143b9c050Sjacobs if ((ptr = strdup(string)) == NULL)
80243b9c050Sjacobs return (PAPI_TEMPORARY_ERROR);
803355b4669Sjacobs
80443b9c050Sjacobs while ((*ptr != '\0') && (result == PAPI_OK)) {
80543b9c050Sjacobs char *key, **values = NULL;
806355b4669Sjacobs
80743b9c050Sjacobs /* strip any leading whitespace */
80843b9c050Sjacobs while (isspace(*ptr) != 0)
80943b9c050Sjacobs ptr++;
81043b9c050Sjacobs
81143b9c050Sjacobs /* Get the name: name[=value] */
81243b9c050Sjacobs key = ptr;
81343b9c050Sjacobs while ((*ptr != '\0') && (*ptr != '=') && (isspace(*ptr) == 0))
81443b9c050Sjacobs ptr++;
81543b9c050Sjacobs
81643b9c050Sjacobs if (*ptr == '=') {
81743b9c050Sjacobs *ptr++ = '\0';
81843b9c050Sjacobs
81943b9c050Sjacobs while ((*ptr != '\0') && (isspace(*ptr) == 0)) {
82043b9c050Sjacobs char *value = ptr;
82143b9c050Sjacobs
82243b9c050Sjacobs if ((*ptr == '\'') || (*ptr == '"')) {
82343b9c050Sjacobs char q = *ptr++;
82443b9c050Sjacobs
82543b9c050Sjacobs /* quoted string value */
82643b9c050Sjacobs while ((*ptr != '\0') && (*ptr != q))
82743b9c050Sjacobs ptr++;
82843b9c050Sjacobs if (*ptr == q)
82943b9c050Sjacobs ptr++;
83043b9c050Sjacobs } else if (*ptr == '{') {
83143b9c050Sjacobs /* collection */
83243b9c050Sjacobs while ((*ptr != '\0') && (*ptr != '}'))
83343b9c050Sjacobs ptr++;
83443b9c050Sjacobs if (*ptr == '}')
83543b9c050Sjacobs ptr++;
836355b4669Sjacobs } else {
83743b9c050Sjacobs /* value */
83843b9c050Sjacobs while ((*ptr != '\0') &&
83943b9c050Sjacobs (*ptr != ',') &&
84043b9c050Sjacobs (isspace(*ptr) == 0))
84143b9c050Sjacobs ptr++;
842355b4669Sjacobs }
84343b9c050Sjacobs if (*ptr == ',')
84443b9c050Sjacobs *ptr++ = '\0';
84543b9c050Sjacobs list_append(&values, value);
846355b4669Sjacobs }
84743b9c050Sjacobs } else { /* boolean "[no]key" */
84843b9c050Sjacobs char *value = "true";
84943b9c050Sjacobs
85043b9c050Sjacobs if (strncasecmp(key, "no", 2) == 0) {
85143b9c050Sjacobs key += 2;
85243b9c050Sjacobs value = "false";
853355b4669Sjacobs }
85443b9c050Sjacobs list_append(&values, value);
855355b4669Sjacobs }
85643b9c050Sjacobs if (*ptr != '\0')
85743b9c050Sjacobs *ptr++ = '\0';
85843b9c050Sjacobs
85943b9c050Sjacobs result = _papiAttributeFromStrings(list, flags, key, values);
86043b9c050Sjacobs free(values);
861355b4669Sjacobs }
862355b4669Sjacobs
863355b4669Sjacobs return (result);
86443b9c050Sjacobs }
865355b4669Sjacobs
866355b4669Sjacobs papi_status_t
papiAttributeListFromString(papi_attribute_t *** attrs,int flags,char * string)8673cf7d3e9SToomas Soome papiAttributeListFromString(papi_attribute_t ***attrs, int flags, char *string)
868355b4669Sjacobs {
869355b4669Sjacobs papi_status_t result = PAPI_OK;
870355b4669Sjacobs
871355b4669Sjacobs if ((attrs != NULL) && (string != NULL) &&
872355b4669Sjacobs ((flags & ~(PAPI_ATTR_APPEND + PAPI_ATTR_REPLACE + PAPI_ATTR_EXCL))
873355b4669Sjacobs == 0)) {
87443b9c050Sjacobs result = _parse_attribute_list(attrs, flags, string);
875355b4669Sjacobs } else {
876355b4669Sjacobs result = PAPI_BAD_ARGUMENT;
877355b4669Sjacobs }
878355b4669Sjacobs
879355b4669Sjacobs return (result);
880355b4669Sjacobs }
881355b4669Sjacobs
882355b4669Sjacobs static papi_status_t
papiAttributeToString(papi_attribute_t * attribute,char * delim,char * buffer,size_t buflen)883355b4669Sjacobs papiAttributeToString(papi_attribute_t *attribute, char *delim,
884355b4669Sjacobs char *buffer, size_t buflen)
885355b4669Sjacobs {
886355b4669Sjacobs papi_attribute_value_t **values = attribute->values;
887355b4669Sjacobs int rc, i;
888355b4669Sjacobs
88943b9c050Sjacobs if ((attribute->type == PAPI_BOOLEAN) && (values[1] == NULL)) {
89043b9c050Sjacobs if (values[0]->boolean == PAPI_FALSE) {
89143b9c050Sjacobs if (isupper(attribute->name[0]) == 0)
89243b9c050Sjacobs strlcat(buffer, "no", buflen);
89343b9c050Sjacobs else
89443b9c050Sjacobs strlcat(buffer, "No", buflen);
89543b9c050Sjacobs }
89643b9c050Sjacobs rc = strlcat(buffer, attribute->name, buflen);
89743b9c050Sjacobs } else {
898355b4669Sjacobs strlcat(buffer, attribute->name, buflen);
89943b9c050Sjacobs rc = strlcat(buffer, "=", buflen);
90043b9c050Sjacobs }
901355b4669Sjacobs
902355b4669Sjacobs if (values == NULL)
903355b4669Sjacobs return (PAPI_OK);
904355b4669Sjacobs
905355b4669Sjacobs for (i = 0; values[i] != NULL; i++) {
906355b4669Sjacobs switch (attribute->type) {
907355b4669Sjacobs case PAPI_STRING:
908355b4669Sjacobs rc = strlcat(buffer, values[i]->string, buflen);
909355b4669Sjacobs break;
910355b4669Sjacobs case PAPI_INTEGER: {
911355b4669Sjacobs char string[24];
912355b4669Sjacobs
913355b4669Sjacobs snprintf(string, sizeof (string), "%d",
914355b4669Sjacobs values[i]->integer);
915355b4669Sjacobs rc = strlcat(buffer, string, buflen);
916355b4669Sjacobs }
917355b4669Sjacobs break;
918355b4669Sjacobs case PAPI_BOOLEAN:
91943b9c050Sjacobs if (values[1] != NULL)
9203cf7d3e9SToomas Soome rc = strlcat(buffer, values[i]->boolean ?
9213cf7d3e9SToomas Soome "true" : "false", buflen);
922355b4669Sjacobs break;
923355b4669Sjacobs case PAPI_RANGE: {
924355b4669Sjacobs char string[24];
925355b4669Sjacobs
92643b9c050Sjacobs if (values[i]->range.lower == values[i]->range.upper)
92743b9c050Sjacobs snprintf(string, sizeof (string), "%d",
92843b9c050Sjacobs values[i]->range.lower);
92943b9c050Sjacobs else
930355b4669Sjacobs snprintf(string, sizeof (string), "%d-%d",
93143b9c050Sjacobs values[i]->range.lower,
93243b9c050Sjacobs values[i]->range.upper);
933355b4669Sjacobs rc = strlcat(buffer, string, buflen);
934355b4669Sjacobs }
935355b4669Sjacobs break;
936355b4669Sjacobs case PAPI_RESOLUTION: {
937355b4669Sjacobs char string[24];
938355b4669Sjacobs
939355b4669Sjacobs snprintf(string, sizeof (string), "%dx%ddp%c",
940355b4669Sjacobs values[i]->resolution.xres,
941355b4669Sjacobs values[i]->resolution.yres,
9423cf7d3e9SToomas Soome values[i]->resolution.units == PAPI_RES_PER_CM ?
9433cf7d3e9SToomas Soome 'c' : 'i');
944355b4669Sjacobs rc = strlcat(buffer, string, buflen);
945355b4669Sjacobs }
946355b4669Sjacobs break;
947355b4669Sjacobs case PAPI_DATETIME: {
948355b4669Sjacobs struct tm *tm = localtime(&values[i]->datetime);
949355b4669Sjacobs
950355b4669Sjacobs if (tm != NULL) {
951355b4669Sjacobs char string[64];
952355b4669Sjacobs
953ee169c7eSGary Mills strftime(string, sizeof (string), "%c", tm);
954355b4669Sjacobs rc = strlcat(buffer, string, buflen);
955355b4669Sjacobs }}
956355b4669Sjacobs break;
957355b4669Sjacobs case PAPI_COLLECTION: {
958355b4669Sjacobs char *string = alloca(buflen);
959355b4669Sjacobs
960355b4669Sjacobs papiAttributeListToString(values[i]->collection,
961355b4669Sjacobs delim, string, buflen);
962355b4669Sjacobs rc = strlcat(buffer, string, buflen);
963355b4669Sjacobs }
964355b4669Sjacobs break;
965355b4669Sjacobs default: {
966355b4669Sjacobs char string[32];
967355b4669Sjacobs
968355b4669Sjacobs snprintf(string, sizeof (string), "unknown-type-0x%x",
969355b4669Sjacobs attribute->type);
970355b4669Sjacobs rc = strlcat(buffer, string, buflen);
971355b4669Sjacobs }
972355b4669Sjacobs }
973355b4669Sjacobs if (values[i+1] != NULL)
974355b4669Sjacobs rc = strlcat(buffer, ",", buflen);
975355b4669Sjacobs
976355b4669Sjacobs if (rc >= buflen)
977355b4669Sjacobs return (PAPI_NOT_POSSIBLE);
978355b4669Sjacobs
979355b4669Sjacobs }
980355b4669Sjacobs
981355b4669Sjacobs return (PAPI_OK);
982355b4669Sjacobs }
983355b4669Sjacobs
984355b4669Sjacobs papi_status_t
papiAttributeListToString(papi_attribute_t ** attrs,char * delim,char * buffer,size_t buflen)985355b4669Sjacobs papiAttributeListToString(papi_attribute_t **attrs,
986355b4669Sjacobs char *delim, char *buffer, size_t buflen)
987355b4669Sjacobs {
988355b4669Sjacobs papi_status_t status = PAPI_OK;
989355b4669Sjacobs int i;
990355b4669Sjacobs
991355b4669Sjacobs if ((attrs == NULL) || (buffer == NULL))
992355b4669Sjacobs return (PAPI_BAD_ARGUMENT);
993355b4669Sjacobs
994355b4669Sjacobs buffer[0] = '\0';
995355b4669Sjacobs if (!delim)
996355b4669Sjacobs delim = " ";
997355b4669Sjacobs
998355b4669Sjacobs for (i = 0; ((attrs[i] != NULL) && (status == PAPI_OK)); i++) {
999355b4669Sjacobs status = papiAttributeToString(attrs[i], delim, buffer, buflen);
1000355b4669Sjacobs if (attrs[i+1] != NULL)
1001355b4669Sjacobs strlcat(buffer, delim, buflen);
1002355b4669Sjacobs }
1003355b4669Sjacobs
1004355b4669Sjacobs return (status);
1005355b4669Sjacobs }
1006355b4669Sjacobs
1007355b4669Sjacobs static int
is_in_list(char * value,char ** list)1008355b4669Sjacobs is_in_list(char *value, char **list)
1009355b4669Sjacobs {
1010355b4669Sjacobs if ((list != NULL) && (value != NULL)) {
1011355b4669Sjacobs int i;
1012355b4669Sjacobs
1013355b4669Sjacobs for (i = 0; list[i] != NULL; i++)
1014355b4669Sjacobs if (strcasecmp(value, list[i]) == 0)
1015355b4669Sjacobs return (0);
1016355b4669Sjacobs }
1017355b4669Sjacobs
1018355b4669Sjacobs return (1);
1019355b4669Sjacobs }
1020355b4669Sjacobs
1021355b4669Sjacobs static papi_status_t
copy_attribute(papi_attribute_t *** list,papi_attribute_t * attribute)1022355b4669Sjacobs copy_attribute(papi_attribute_t ***list, papi_attribute_t *attribute)
1023355b4669Sjacobs {
1024355b4669Sjacobs papi_status_t status;
1025355b4669Sjacobs int i = 0;
1026355b4669Sjacobs
1027355b4669Sjacobs if ((list == NULL) || (attribute == NULL) ||
1028355b4669Sjacobs (attribute->values == NULL))
1029355b4669Sjacobs return (PAPI_BAD_ARGUMENT);
1030355b4669Sjacobs
1031355b4669Sjacobs for (status = papiAttributeListAddValue(list, PAPI_ATTR_EXCL,
10323cf7d3e9SToomas Soome attribute->name, attribute->type, attribute->values[i]);
1033355b4669Sjacobs ((status == PAPI_OK) && (attribute->values[i] != NULL));
1034355b4669Sjacobs status = papiAttributeListAddValue(list, PAPI_ATTR_APPEND,
10353cf7d3e9SToomas Soome attribute->name, attribute->type, attribute->values[i]))
1036355b4669Sjacobs i++;
1037355b4669Sjacobs
1038355b4669Sjacobs return (status);
1039355b4669Sjacobs }
1040355b4669Sjacobs
1041355b4669Sjacobs void
copy_attributes(papi_attribute_t *** result,papi_attribute_t ** attributes)1042355b4669Sjacobs copy_attributes(papi_attribute_t ***result, papi_attribute_t **attributes)
1043355b4669Sjacobs {
1044355b4669Sjacobs int i;
1045355b4669Sjacobs
1046355b4669Sjacobs if ((result == NULL) || (attributes == NULL))
1047355b4669Sjacobs return;
1048355b4669Sjacobs
1049355b4669Sjacobs for (i = 0; attributes[i] != NULL; i++)
1050355b4669Sjacobs copy_attribute(result, attributes[i]);
1051355b4669Sjacobs }
1052355b4669Sjacobs
1053355b4669Sjacobs void
split_and_copy_attributes(char ** list,papi_attribute_t ** attributes,papi_attribute_t *** in,papi_attribute_t *** out)1054355b4669Sjacobs split_and_copy_attributes(char **list, papi_attribute_t **attributes,
1055355b4669Sjacobs papi_attribute_t ***in, papi_attribute_t ***out)
1056355b4669Sjacobs {
1057355b4669Sjacobs int i;
1058355b4669Sjacobs
1059355b4669Sjacobs if ((list == NULL) || (attributes == NULL))
1060355b4669Sjacobs return;
1061355b4669Sjacobs
1062355b4669Sjacobs for (i = 0; attributes[i] != NULL; i++)
1063355b4669Sjacobs if (is_in_list(attributes[i]->name, list) == 0)
1064355b4669Sjacobs copy_attribute(in, attributes[i]);
1065355b4669Sjacobs else
1066355b4669Sjacobs copy_attribute(out, attributes[i]);
1067355b4669Sjacobs }
1068355b4669Sjacobs
1069355b4669Sjacobs void
papiAttributeListPrint(FILE * fp,papi_attribute_t ** attributes,char * prefix_fmt,...)1070355b4669Sjacobs papiAttributeListPrint(FILE *fp, papi_attribute_t **attributes,
1071355b4669Sjacobs char *prefix_fmt, ...)
1072355b4669Sjacobs {
1073355b4669Sjacobs char *prefix = NULL;
1074355b4669Sjacobs char *buffer = NULL;
1075355b4669Sjacobs char *newfmt = NULL;
1076355b4669Sjacobs void *mem;
1077355b4669Sjacobs ssize_t size = 0;
1078355b4669Sjacobs va_list ap;
1079355b4669Sjacobs
1080355b4669Sjacobs newfmt = malloc(strlen(prefix_fmt) + 2);
1081355b4669Sjacobs sprintf(newfmt, "\n%s", prefix_fmt);
1082355b4669Sjacobs
1083355b4669Sjacobs va_start(ap, prefix_fmt);
1084355b4669Sjacobs while (vsnprintf(prefix, size, newfmt, ap) > size) {
1085355b4669Sjacobs size += 1024;
1086355b4669Sjacobs mem = realloc(prefix, size);
1087355b4669Sjacobs if (!mem) goto error;
1088355b4669Sjacobs prefix = mem;
1089355b4669Sjacobs }
1090355b4669Sjacobs va_end(ap);
1091355b4669Sjacobs
1092355b4669Sjacobs if (attributes) {
1093355b4669Sjacobs size = 0;
1094355b4669Sjacobs while (papiAttributeListToString(attributes, prefix, buffer,
1095355b4669Sjacobs size) != PAPI_OK) {
1096355b4669Sjacobs size += 1024;
1097355b4669Sjacobs mem = realloc(buffer, size);
1098355b4669Sjacobs if (!mem) goto error;
1099355b4669Sjacobs buffer = mem;
1100355b4669Sjacobs }
1101355b4669Sjacobs }
1102355b4669Sjacobs
1103355b4669Sjacobs fprintf(fp, "%s%s\n", prefix, buffer ? buffer : "");
1104355b4669Sjacobs fflush(fp);
1105355b4669Sjacobs
1106355b4669Sjacobs error:
1107355b4669Sjacobs free(newfmt);
1108355b4669Sjacobs free(prefix);
1109355b4669Sjacobs free(buffer);
1110355b4669Sjacobs }
1111