script_uboot.c 5.86 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
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
/*
 * 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 2 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/>.
 */
#include "common.h"

#include <stdint.h>
#include <stdio.h>
#include <string.h>

#include "script.h"
#include "script_uboot.h"

#define pr_info(...)	errf("fexc-uboot: " __VA_ARGS__)
#define pr_err(...)	errf("E: fexc-uboot: " __VA_ARGS__)

#ifdef DEBUG
#define pr_debug(...)	errf("D: fexc-uboot: " __VA_ARGS__)
#else
#define pr_debug(...)
#endif

35
36
37
38
39
struct members {
	const char *name;
	const char *translation;
	int mode;
};
40
41
#define foreach_member(I, T) for (const struct members *I = T; \
	     I < T+ARRAY_SIZE(T); I++)
42
43
44

/*
 */
Alejandro Mery's avatar
Alejandro Mery committed
45
46
static inline void out_u32_member(FILE *out, const char *key, int hexa,
				  struct script_single_entry *val)
Alejandro Mery's avatar
Alejandro Mery committed
47
48
{
	const char *fmt;
49
	if (hexa)
Alejandro Mery's avatar
Alejandro Mery committed
50
51
52
53
		fmt = "\t.%s = %#x,\n";
	else
		fmt = "\t.%s = %u,\n";

Alejandro Mery's avatar
Alejandro Mery committed
54
	fprintf(out, fmt, key, val->value);
Alejandro Mery's avatar
Alejandro Mery committed
55
56
}

57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
static inline void out_gpio_member(FILE *out, const char *key,
				   struct script_gpio_entry *gpio)
{
	fprintf(out, "\t.%s = ", key);

	if (gpio->port == 0xffff)
		fprintf(out, "GPIO_AXP_CFG(%u", gpio->port_num);
	else
		fprintf(out, "GPIO_CFG(%u, %u", gpio->port, gpio->port_num);

	for (const int *p = gpio->data, *pe = p+4; p != pe; p++) {
		if (*p == -1)
			fputs(", 0xff", out);
		else
			fprintf(out, ", %u", *p);
	}

	fputs("),\n", out);
}

Alejandro Mery's avatar
Alejandro Mery committed
77
78
79
80
81
static inline void out_null_member(FILE *out, const char *key)
{
	fprintf(out, "\t/* %s is NULL */\n", key);
}

82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
static inline int out_member(FILE *out, const char *key, int mode,
		      struct script_entry *ep)
{
	switch (ep->type) {
	case SCRIPT_VALUE_TYPE_SINGLE_WORD:
		out_u32_member(out, key, mode,
		       container_of(ep, struct script_single_entry, entry));
		break;
	case SCRIPT_VALUE_TYPE_NULL:
		out_null_member(out, key);
		break;
	case SCRIPT_VALUE_TYPE_GPIO:
		out_gpio_member(out, key,
			container_of(ep, struct script_gpio_entry, entry));
		break;
	default:
		return 0;
	}
	return 1;
}

103
104
105
/*
 * DRAM
 */
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
static struct members dram_members[] = {
	{ .name="dram_clock" },
	{ .name="dram_clk", .translation="clock" },
	{ .name="dram_type" },
	{ .name="dram_rank_num" },
	{ .name="dram_density" },
	{ .name="dram_chip_density", .translation="density" },
	{ .name="dram_io_width" },
	{ .name="dram_bus_width" },
	{ .name="dram_cas" },
	{ .name="dram_zq" },
	{ .name="dram_odt_en" },
	{ .name="dram_size" },
	{ .name="dram_tpr0", .mode=1 },
	{ .name="dram_tpr1", .mode=1 },
	{ .name="dram_tpr2", .mode=1 },
	{ .name="dram_tpr3", .mode=1 },
	{ .name="dram_tpr4", .mode=1 },
	{ .name="dram_tpr5", .mode=1 },
	{ .name="dram_emr1", .mode=1 },
	{ .name="dram_emr2", .mode=1 },
	{ .name="dram_emr3", .mode=1 },
};

Alejandro Mery's avatar
Alejandro Mery committed
130
131
132
133
static int generate_dram_struct(FILE *out, struct script_section *sp)
{
	struct script_entry *ep;
	const char *key;
134
	int ret = 1;
Alejandro Mery's avatar
Alejandro Mery committed
135

136
	fprintf(out, "static struct dram_para dram_para = {\n");
137
	foreach_member(mp, dram_members) {
138
139
140
141
142
143
		ep = script_find_entry(sp, mp->name);
		if (!ep)
			continue;

		key = (mp->translation) ? mp->translation : mp->name+5;
		if (!out_member(out, key, mp->mode, ep)) {
Alejandro Mery's avatar
Alejandro Mery committed
144
145
146
147
148
149
			pr_err("dram_para: %s: invalid field\n", ep->name);
			ret = 0;
		}

	}
	fprintf(out, "};\n");
150
151
152
153
154
155
156
	fputs("\nint sunxi_dram_init(void)\n"
	      "{\n\treturn DRAMC_init(&dram_para);\n}\n",
	      out);

	return ret;
}

157
#if 0
158
159
160
/*
 * PMU
 */
161
162
163
164
165
166
167
168
169
170
static struct members pmu_members[] = {
	{ .name = "pmu_used2" },
	{ .name = "pmu_para" },
	{ .name = "pmu_adpdet" },
	{ .name = "pmu_shutdown_chgcur" },
	{ .name = "pmu_shutdown_chgcur2" },
	{ .name = "pmu_pwroff_vol" },
	{ .name = "pmu_pwron_vol" },
};

171
172
static int generate_pmu_struct(FILE *out, struct script_section *target,
			       struct script_section *pmu_para)
173
174
{
	struct list_entry *le;
175
	struct script_section *sp;
176
	struct script_entry *ep;
177
	const char *key;
178
179
180
	int ret = 1;

	fputs("\nstatic struct pmu_para pmu_para = {\n", out);
181
182

	sp = target;
183
184
185
186
	for (le = list_first(&sp->entries); le;
	     le = list_next(&sp->entries, le)) {
		ep = container_of(le, struct script_entry, entries);

187
		if (!out_member(out, ep->name, 0, ep)) {
188
			pr_err("target: %s: invalid field\n", ep->name);
189
190
191
			ret = 0;
		}
	}
192

193
	foreach_member(mp, pmu_members) {
194
195
		ep = script_find_entry(pmu_para, mp->name);
		if (!ep)
196
			continue;
197
198
199
200
201

		key = (mp->translation) ? mp->translation : mp->name+4;
		if (!out_member(out, key, mp->mode, ep)) {
			pr_err("pmu_para: %s: invalid field\n", mp->name);
			ret = 0;
202
203
204
		}
	}

205
206
207
208
	fputs("};\n", out);
	fputs("\nint sunxi_pmu_init(void)\n"
	      "{\n\treturn PMU_init(&pmu_para);\n}\n",
	      out);
Alejandro Mery's avatar
Alejandro Mery committed
209
	return ret;
210
211

	(void) pmu_para;
Alejandro Mery's avatar
Alejandro Mery committed
212
}
213
#endif
Alejandro Mery's avatar
Alejandro Mery committed
214
215
216
217

int script_generate_uboot(FILE *out, const char *UNUSED(filename),
			  struct script *script)
{
218
219
220
221
222
	struct {
		const char *name;
		struct script_section *sp;
	} sections[] = {
		{ "dram_para", NULL },
223
#if 0
224
225
		{ "target", NULL },
		{ "pmu_para", NULL },
226
#endif
227
228
229
230
231
232
233
234
235
236
237
238
239
240
	};

	for (unsigned i=0; i<ARRAY_SIZE(sections); i++) {
		struct script_section *sp;

		sp = script_find_section(script, sections[i].name);
		if (sp)
			sections[i].sp = sp;
		else {
			pr_err("%s: critical section missing",
			       sections[i].name);
			return 0;
		}
	}
Alejandro Mery's avatar
Alejandro Mery committed
241

242
243
	fputs("/* this file is generated, don't edit it yourself */\n\n"
	      "#include <common.h>\n"
244
#if 0
245
	      "#include <asm/arch/pmu.h>\n"
246
#endif
247
248
	      "#include <asm/arch/dram.h>\n\n",
	      out);
Alejandro Mery's avatar
Alejandro Mery committed
249

250
	generate_dram_struct(out, sections[0].sp);
251
#if 0
252
	generate_pmu_struct(out, sections[1].sp, sections[2].sp);
253
#endif
254

Alejandro Mery's avatar
Alejandro Mery committed
255
256
	return 1;
}