1 /* $NetBSD: citrus_pivot_factory.c,v 1.7 2009/04/12 14:20:19 lukem Exp $ */
4 * Copyright (c)2003 Citrus Project,
7 * Redistribution and use in source and binary forms, with or without
8 * modification, are permitted provided that the following conditions
10 * 1. Redistributions of source code must retain the above copyright
11 * notice, this list of conditions and the following disclaimer.
12 * 2. Redistributions in binary form must reproduce the above copyright
13 * notice, this list of conditions and the following disclaimer in the
14 * documentation and/or other materials provided with the distribution.
16 * THIS SOFTWARE IS PROVIDED BY THE AUTHOR AND CONTRIBUTORS ``AS IS'' AND
17 * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
18 * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
19 * ARE DISCLAIMED. IN NO EVENT SHALL THE AUTHOR OR CONTRIBUTORS BE LIABLE
20 * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
21 * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
22 * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
23 * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
24 * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
25 * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
29 #if HAVE_NBTOOL_CONFIG_H
30 #include "nbtool_config.h"
33 #include <sys/cdefs.h>
34 #if defined(LIBC_SCCS) && !defined(lint)
35 __RCSID("$NetBSD: citrus_pivot_factory.c,v 1.7 2009/04/12 14:20:19 lukem Exp $");
36 #endif /* LIBC_SCCS and not lint */
45 #include <sys/queue.h>
47 #include "citrus_namespace.h"
48 #include "citrus_region.h"
49 #include "citrus_bcs.h"
50 #include "citrus_db_factory.h"
51 #include "citrus_db_hash.h"
52 #include "citrus_pivot_file.h"
53 #include "citrus_pivot_factory.h"
57 struct _citrus_db_factory
*se_df
;
58 SIMPLEQ_ENTRY(src_entry
) se_entry
;
60 SIMPLEQ_HEAD(src_head
, src_entry
);
63 find_src(struct src_head
*sh
, struct src_entry
**rse
, const char *name
)
68 SIMPLEQ_FOREACH(se
, sh
, se_entry
) {
69 if (_bcs_strcasecmp(se
->se_name
, name
) == 0) {
74 se
= malloc(sizeof(*se
));
77 se
->se_name
= strdup(name
);
78 if (se
->se_name
== NULL
) {
83 ret
= _db_factory_create(&se
->se_df
, &_db_hash_std
, NULL
);
89 SIMPLEQ_INSERT_TAIL(sh
, se
, se_entry
);
96 free_src(struct src_head
*sh
)
100 while ((se
= SIMPLEQ_FIRST(sh
)) != NULL
) {
101 SIMPLEQ_REMOVE_HEAD(sh
, se_entry
);
102 _db_factory_free(se
->se_df
);
111 convert_line(struct src_head
*sh
, const char *line
, size_t len
)
114 struct src_entry
*se
;
116 char key1
[LINE_MAX
], key2
[LINE_MAX
], data
[LINE_MAX
];
120 se
= NULL
; /* XXX gcc */
122 /* cut off trailing comment */
123 p
= memchr(line
, T_COMM
, len
);
128 line
= _bcs_skip_ws_len(line
, &len
);
131 p
= _bcs_skip_nonws_len(line
, &len
);
134 snprintf(key1
, sizeof(key1
), "%.*s", (int)(p
-line
), line
);
137 line
= _bcs_skip_ws_len(p
, &len
);
140 p
= _bcs_skip_nonws_len(line
, &len
);
143 snprintf(key2
, sizeof(key2
), "%.*s", (int)(p
-line
), line
);
146 line
= _bcs_skip_ws_len(p
, &len
);
147 _bcs_trunc_rws_len(line
, &len
);
148 snprintf(data
, sizeof(data
), "%.*s", (int)len
, line
);
149 val
= strtoul(data
, &ep
, 0);
154 ret
= find_src(sh
, &se
, key1
);
158 return _db_factory_add32_by_s(se
->se_df
, key2
, val
);
162 dump_db(struct src_head
*sh
, struct _region
*r
)
165 struct _db_factory
*df
;
166 struct src_entry
*se
;
171 ret
= _db_factory_create(&df
, &_db_hash_std
, NULL
);
175 SIMPLEQ_FOREACH(se
, sh
, se_entry
) {
176 size
= _db_factory_calc_size(se
->se_df
);
180 _region_init(&subr
, ptr
, size
);
181 ret
= _db_factory_serialize(se
->se_df
, _CITRUS_PIVOT_SUB_MAGIC
,
185 ret
= _db_factory_add_by_s(df
, se
->se_name
, &subr
, 1);
190 size
= _db_factory_calc_size(df
);
194 _region_init(r
, ptr
, size
);
196 ret
= _db_factory_serialize(df
, _CITRUS_PIVOT_MAGIC
, r
);
201 _db_factory_free(df
);
206 _citrus_pivot_factory_convert(FILE *out
, FILE *in
)
216 while ((line
= fgetln(in
, &size
)) != NULL
)
217 if ((ret
= convert_line(&sh
, line
, size
))) {
222 ret
= dump_db(&sh
, &r
);
227 if (fwrite(_region_head(&r
), _region_size(&r
), 1, out
) != 1)