script_fex.c 7.85 KB
Newer Older
Alejandro Mery's avatar
Alejandro Mery committed
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
/*
 * Copyright (C) 2012  Alejandro Mery <amery@geeks.cl>
 *
 * This program is free software: you can redistribute it and/or modify
 * it under the terms of the GNU General Public License as published by
 * the Free Software Foundation, either version 3 of the License, or
 * (at your option) any later version.
 *
 * This program is distributed in the hope that it will be useful,
 * but WITHOUT ANY WARRANTY; without even the implied warranty of
 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
 * GNU General Public License for more details.
 *
 * You should have received a copy of the GNU General Public License
 * along with this program.  If not, see <http://www.gnu.org/licenses/>.
 */
17
#include "common.h"
Alejandro Mery's avatar
Alejandro Mery committed
18

19
#include <ctype.h>
20
#include <stdint.h>
21
#include <stdio.h>
22
#include <stdlib.h>
23
24
#include <string.h>

25
26
#include "script.h"
#include "script_fex.h"
27

28
#define MAX_LINE	255
29

30
31
32
#define pr_info(...)	fprintf(stderr, "fex2bin: " __VA_ARGS__)
#define pr_err(...)	pr_info("E: " __VA_ARGS__)

33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
/*
 * generator
 */
static inline size_t strlen2(const char *s)
{
	size_t l = strlen(s);
	const char *p = &s[l-1];
	while (l && *p >= '0' && *p <= '9') {
		l--;
		p--;
	}
	return l;
}

static int find_full_match(const char *s, size_t l, const char **list)
{
	while (*list) {
		if (memcmp(s, *list, l) == 0)
			return 1;
		list++;
	}

	return 0;
}

/**
 */
static int decompile_single_mode(const char *name)
{
	static const char *hexa_entries[] = {
		"dram_baseaddr", "dram_zq", "dram_tpr", "dram_emr",
		"g2d_size",
		"rtp_press_threshold", "rtp_sensitive_level",
		"ctp_twi_addr", "csi_twi_addr", "csi_twi_addr_b", "tkey_twi_addr",
		"lcd_gamma_tbl_",
		"gsensor_twi_addr",
		NULL };
	size_t l = strlen2(name);

	if (find_full_match(name, l, hexa_entries))
		return 0;
	else
		return -1;
}

int script_generate_fex(FILE *out, const char *UNUSED(filename),
			struct script *script)
{
	struct list_entry *ls, *le;
	struct script_section *section;
	struct script_entry *entry;

	for (ls = list_first(&script->sections); ls;
	     ls = list_next(&script->sections, ls)) {
		section = container_of(ls, struct script_section, sections);

		fprintf(out, "[%s]\n", section->name);
		for (le = list_first(&section->entries); le;
		     le = list_next(&section->entries, le)) {
			entry = container_of(le, struct script_entry, entries);

			switch(entry->type) {
			case SCRIPT_VALUE_TYPE_SINGLE_WORD: {
				int mode = decompile_single_mode(entry->name);
				struct script_single_entry *single;
				single = container_of(entry, struct script_single_entry, entry);

100
				fprintf(out, "%s = ", entry->name);
101
102
103
104
105
106
107
108
109
110
111
				if (mode < 0)
					fprintf(out, "%d", single->value);
				else if (mode > 0)
					fprintf(out, "0x%0*x", mode, single->value);
				else
					fprintf(out, "0x%x", single->value);
				fputc('\n', out);
				}; break;
			case SCRIPT_VALUE_TYPE_STRING: {
				struct script_string_entry *string;
				string = container_of(entry, struct script_string_entry, entry);
112
				fprintf(out, "%s = \"%.*s\"\n", entry->name,
113
114
115
116
117
118
119
120
121
122
					(int)string->l, string->string);
				}; break;
			case SCRIPT_VALUE_TYPE_MULTI_WORD:
				abort();
			case SCRIPT_VALUE_TYPE_GPIO: {
				char port = 'A'-1;
				struct script_gpio_entry *gpio;
				gpio = container_of(entry, struct script_gpio_entry, entry);

				port += gpio->port;
123
				fprintf(out, "%s = port:P%c%02d", entry->name, port, gpio->port_num);
124
125
126
127
128
129
130
131
132
				for (const int *p = gpio->data, *pe = p+4; p != pe; p++) {
					if (*p == -1)
						fputs("<default>", out);
					else
						fprintf(out, "<%d>", *p);
				}
				fputc('\n', out);
				}; break;
			case SCRIPT_VALUE_TYPE_NULL:
133
				fprintf(out, "%s =\n", entry->name);
134
135
136
137
138
139
140
141
142
143
144
145
				break;
			}
		}
		fputc('\n', out);
	}
	return 0;
}

/*
 * parser
 */

146
147
148
149
150
151
152
153
154
155
/** find first not blank char */
static inline char *skip_blank(char *p)
{
	while(isblank(*p))
		p++;
	return p;
}

/** trim out blank chars at the end of a string */
static inline char *rtrim(const char *s, char *p)
156
{
157
158
159
160
161
162
	if (p>s) {
		while (p!=s && isblank(*--p))
			;
		*++p='\0';
	}
	return p;
163
164
165
166
}

/**
 */
167
int script_parse_fex(FILE *in, const char *filename, struct script *script)
168
169
170
{
	char buffer[MAX_LINE+1];
	int ok = 1;
171
	struct script_section *last_section = NULL;
172

173
	/* TODO: deal with longer lines correctly (specially in comments) */
174
	for(size_t line = 1; ok && fgets(buffer, sizeof(buffer), in); line++) {
175
176
		char *s = skip_blank(buffer); /* beginning */
		char *pe = s; /* \0... to be found */
177

178
		if (*pe) while (*++pe)
179
			;
180
181
182
183
184
185
186
187
188
189
190
191

		if (pe>s && pe[-1] == '\n') {
			if (pe>s+1 && pe[-2] == '\r')
				pe -= 2;
			else
				pe -= 1;
			*pe = '\0';
		}

		pe = rtrim(s, pe);

		if (pe == s || *s == ';' || *s == '#')
192
			continue; /* empty */
193
194
195
196
197
198
199
200
		else if (*s == '[') {
			/* section */
			char *p = ++s;
			while (isalnum(*p) || *p == '_')
				p++;

			if (*p == ']' && *(p+1) == '\0') {
				*p = '\0';
201
202
203
204
				if ((last_section = script_section_new(script, s)))
					continue;

				perror("malloc");
205
206
207
			} else if (*p) {
				errf("E: %s:%zu: invalid character at %zu.\n",
				     filename, line, p-buffer+1);
208
209
210
			} else {
				errf("E: %s:%zu: incomplete section declaration.\n",
				     filename, line);
211
			}
212
			ok = 0;
213
214
		} else {
			/* key = value */
215
216
217
			const char *key = s;
			char *mark, *p = s;

218
219
220
			if (!last_section) {
				errf("E: %s:%zu: data must follow a section.\n",
				     filename, line);
221
				goto parse_error;
222
223
			};

224
225
226
227
			while (isalnum(*p) || *p == '_')
				p++;
			mark = p;
			p = skip_blank(p);
228
229
			if (*p != '=')
				goto invalid_char_at_p;
230
231
232
233
234
			*mark = '\0'; /* truncate key */
			p = skip_blank(p+1);

			if (*p == '\0') {
				/* NULL */
235
236
237
				if (script_null_entry_new(last_section, key))
					continue;
				perror("malloc");
238
239
240
			} else if (pe > p+1 && *p == '"' && pe[-1] == '"') {
				/* string */
				p++; *--pe = '\0';
241
				if (script_string_entry_new(last_section, key, pe-p, p)) {
242
#ifdef VERBOSE
243
244
245
					errf("%s.%s = \"%.*s\"\n",
					     last_section->name, key,
					     (int)(pe-p), p);
246
#endif
247
					continue;
248
				}
249
				perror("malloc");
250
251
252
			} else if (memcmp("port:P", p, 6) == 0) {
				/* GPIO */
				p += 6;
Alejandro Mery's avatar
Alejandro Mery committed
253
254
255
256
				if (*p < 'A' || *p > 'Z')
					;
				else {
					char *end;
257
					int port = *p++ - 'A' + 1;
258
					long v = strtol(p, &end, 10);
Alejandro Mery's avatar
Alejandro Mery committed
259
					if (end == p)
260
261
262
263
						goto invalid_char_at_p;
					else if (v<0 || v>255) {
						errf("E: %s:%zu: port out of range at %zu (%ld).\n",
						     filename, line, p-buffer+1, v);
Alejandro Mery's avatar
Alejandro Mery committed
264
					} else {
265
266
						int data[] = {-1,-1,-1,-1};
						int port_num = v;
Alejandro Mery's avatar
Alejandro Mery committed
267
						p = end;
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
						for (int i=0; *p && i<4; i++) {
							if (memcmp(p, "<default>", 9) == 0) {
								p += 9;
								continue;
							} else if (*p == '<') {
								v = strtol(++p, &end, 10);
								if (end == p) {
									;
								} else if (v<0 || v>INT32_MAX) {
									errf("E: %s:%zu: value out of range at %zu (%ld).\n",
									     filename, line, p-buffer+1, v);
									goto parse_error;
								} else if (*end != '>') {
									p = end;
								} else {
									p = end+1;
									data[i] = v;
									continue;
								}
							}
							break;
						}
						if (*p)
							goto invalid_char_at_p;
						if (script_gpio_entry_new(last_section, key,
									  port, port_num, data)) {
294
#ifdef VERBOSE
295
296
297
298
							errf("%s.%s = GPIO %d.%d (%d,%d,%d,%d)\n",
							     last_section->name, key,
							     port, port_num,
							     data[0], data[1], data[2], data[3]);
299
#endif
300
301
302
							continue;
						}
						perror("malloc");
Alejandro Mery's avatar
Alejandro Mery committed
303
304
					}
				}
305
306
307
308
			} else if (isdigit(*p)) {
				long long v = 0;
				char *end;
				v = strtoll(p, &end, 0);
309
310
				p = end;
				if (p != pe) {
311
					goto invalid_char_at_p;
312
313
314
				} else if (v > UINT32_MAX) {
					errf("E: %s:%zu: value out of range %lld.\n",
					     filename, line, v);
315
				} else if (script_single_entry_new(last_section, key, v)) {
316
#ifdef VERBOSE
317
318
					errf("%s.%s = %lld\n",
					     last_section->name, key, v);
319
#endif
320
					continue;
321
				}
322
			}
323
			goto parse_error;
324
325
326
327
invalid_char_at_p:
			errf("E: %s:%zu: invalid character at %zu.\n",
			     filename, line, p-buffer+1);
parse_error:
328
			ok = 0;
329
		}
330
331
	};

332
	if (ferror(in))
333
334
335
		ok = 0;
	return ok;
}