socfpga_sip_svc.c 13.2 KB
Newer Older
1
/*
2
 * Copyright (c) 2019-2020, ARM Limited and Contributors. All rights reserved.
3
4
5
6
7
8
9
 *
 * SPDX-License-Identifier: BSD-3-Clause
 */

#include <assert.h>
#include <common/debug.h>
#include <common/runtime_svc.h>
10
#include <lib/mmio.h>
11
12
#include <tools_share/uuid.h>

13
#include "socfpga_mailbox.h"
14
#include "socfpga_reset_manager.h"
15
#include "socfpga_sip_svc.h"
16
17
18
19
20
21
22

/* Number of SiP Calls implemented */
#define SIP_NUM_CALLS		0x3

/* Total buffer the driver can hold */
#define FPGA_CONFIG_BUFFER_SIZE 4

23
24
25
26
27
28
29
30
static int current_block;
static int read_block;
static int current_buffer;
static int send_id;
static int rcv_id;
static int max_blocks;
static uint32_t bytes_per_block;
static uint32_t blocks_submitted;
31
static int is_partial_reconfig;
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46

struct fpga_config_info {
	uint32_t addr;
	int size;
	int size_written;
	uint32_t write_requested;
	int subblocks_sent;
	int block_number;
};

/*  SiP Service UUID */
DEFINE_SVC_UUID2(intl_svc_uid,
		0xa85273b0, 0xe85a, 0x4862, 0xa6, 0x2a,
		0xfa, 0x88, 0x88, 0x17, 0x68, 0x81);

47
static uint64_t socfpga_sip_handler(uint32_t smc_fid,
48
49
50
51
52
53
54
55
56
57
58
59
60
61
				   uint64_t x1,
				   uint64_t x2,
				   uint64_t x3,
				   uint64_t x4,
				   void *cookie,
				   void *handle,
				   uint64_t flags)
{
	ERROR("%s: unhandled SMC (0x%x)\n", __func__, smc_fid);
	SMC_RET1(handle, SMC_UNK);
}

struct fpga_config_info fpga_config_buffers[FPGA_CONFIG_BUFFER_SIZE];

Hadi Asyrafi's avatar
Hadi Asyrafi committed
62
static int intel_fpga_sdm_write_buffer(struct fpga_config_info *buffer)
63
{
64
	uint32_t args[3];
65
66

	while (max_blocks > 0 && buffer->size > buffer->size_written) {
Hadi Asyrafi's avatar
Hadi Asyrafi committed
67
68
69
		args[0] = (1<<8);
		args[1] = buffer->addr + buffer->size_written;
		if (buffer->size - buffer->size_written <= bytes_per_block) {
70
71
72
			args[2] = buffer->size - buffer->size_written;
			current_buffer++;
			current_buffer %= FPGA_CONFIG_BUFFER_SIZE;
Hadi Asyrafi's avatar
Hadi Asyrafi committed
73
		} else
74
			args[2] = bytes_per_block;
Hadi Asyrafi's avatar
Hadi Asyrafi committed
75
76

		buffer->size_written += args[2];
77
78
79
		mailbox_send_cmd_async(send_id++ % MBOX_MAX_JOB_ID,
					MBOX_RECONFIG_DATA, args, 3,
					CMD_CASUAL, CMD_INDIRECT);
Hadi Asyrafi's avatar
Hadi Asyrafi committed
80
81

		buffer->subblocks_sent++;
82
83
		max_blocks--;
	}
Hadi Asyrafi's avatar
Hadi Asyrafi committed
84
85

	return !max_blocks;
86
87
88
89
}

static int intel_fpga_sdm_write_all(void)
{
Hadi Asyrafi's avatar
Hadi Asyrafi committed
90
91
92
93
	for (int i = 0; i < FPGA_CONFIG_BUFFER_SIZE; i++)
		if (intel_fpga_sdm_write_buffer(
			&fpga_config_buffers[current_buffer]))
			break;
94
95
96
	return 0;
}

97
static uint32_t intel_mailbox_fpga_config_isdone(uint32_t query_type)
98
{
99
100
101
102
103
104
	uint32_t ret;

	if (query_type == 1)
		ret = intel_mailbox_get_config_status(MBOX_CONFIG_STATUS);
	else
		ret = intel_mailbox_get_config_status(MBOX_RECONFIG_STATUS);
Hadi Asyrafi's avatar
Hadi Asyrafi committed
105
106
107
108
109
110
111
112

	if (ret) {
		if (ret == MBOX_CFGSTAT_STATE_CONFIG)
			return INTEL_SIP_SMC_STATUS_BUSY;
		else
			return INTEL_SIP_SMC_STATUS_ERROR;
	}

113
114
115
116
117
118
	if (query_type != 1) {
		/* full reconfiguration */
		if (!is_partial_reconfig)
			socfpga_bridges_enable();	/* Enable bridge */
	}

Hadi Asyrafi's avatar
Hadi Asyrafi committed
119
	return INTEL_SIP_SMC_STATUS_OK;
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
}

static int mark_last_buffer_xfer_completed(uint32_t *buffer_addr_completed)
{
	int i;

	for (i = 0; i < FPGA_CONFIG_BUFFER_SIZE; i++) {
		if (fpga_config_buffers[i].block_number == current_block) {
			fpga_config_buffers[i].subblocks_sent--;
			if (fpga_config_buffers[i].subblocks_sent == 0
			&& fpga_config_buffers[i].size <=
			fpga_config_buffers[i].size_written) {
				fpga_config_buffers[i].write_requested = 0;
				current_block++;
				*buffer_addr_completed =
					fpga_config_buffers[i].addr;
				return 0;
			}
		}
	}

	return -1;
}

144
static int intel_fpga_config_completed_write(uint32_t *completed_addr,
145
146
147
148
149
150
151
152
					uint32_t *count)
{
	uint32_t status = INTEL_SIP_SMC_STATUS_OK;
	*count = 0;
	int resp_len = 0;
	uint32_t resp[5];
	int all_completed = 1;

153
	while (*count < 3) {
154

155
		resp_len = mailbox_read_response(rcv_id % MBOX_MAX_JOB_ID,
156
				resp, ARRAY_SIZE(resp));
157

158
159
		if (resp_len < 0)
			break;
160
161

		max_blocks++;
162
163
		rcv_id++;

164
165
166
167
168
169
170
171
172
173
		if (mark_last_buffer_xfer_completed(
			&completed_addr[*count]) == 0)
			*count = *count + 1;
		else
			break;
	}

	if (*count <= 0) {
		if (resp_len != MBOX_NO_RESPONSE &&
			resp_len != MBOX_TIMEOUT && resp_len != 0) {
174
			mailbox_clear_response();
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
			return INTEL_SIP_SMC_STATUS_ERROR;
		}

		*count = 0;
	}

	intel_fpga_sdm_write_all();

	if (*count > 0)
		status = INTEL_SIP_SMC_STATUS_OK;
	else if (*count == 0)
		status = INTEL_SIP_SMC_STATUS_BUSY;

	for (int i = 0; i < FPGA_CONFIG_BUFFER_SIZE; i++) {
		if (fpga_config_buffers[i].write_requested != 0) {
			all_completed = 0;
			break;
		}
	}

	if (all_completed == 1)
		return INTEL_SIP_SMC_STATUS_OK;

	return status;
}

201
static int intel_fpga_config_start(uint32_t config_type)
202
203
204
205
{
	uint32_t response[3];
	int status = 0;

206
207
	is_partial_reconfig = config_type;

208
209
	mailbox_clear_response();

210
	mailbox_send_cmd(1, MBOX_CMD_CANCEL, NULL, 0, CMD_CASUAL, NULL, 0);
211

212
213
	status = mailbox_send_cmd(1, MBOX_RECONFIG, NULL, 0, CMD_CASUAL,
			response, ARRAY_SIZE(response));
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231

	if (status < 0)
		return status;

	max_blocks = response[0];
	bytes_per_block = response[1];

	for (int i = 0; i < FPGA_CONFIG_BUFFER_SIZE; i++) {
		fpga_config_buffers[i].size = 0;
		fpga_config_buffers[i].size_written = 0;
		fpga_config_buffers[i].addr = 0;
		fpga_config_buffers[i].write_requested = 0;
		fpga_config_buffers[i].block_number = 0;
		fpga_config_buffers[i].subblocks_sent = 0;
	}

	blocks_submitted = 0;
	current_block = 0;
232
	read_block = 0;
233
	current_buffer = 0;
234
235
	send_id = 0;
	rcv_id = 0;
236

237
238
239
240
241
242
	/* full reconfiguration */
	if (!is_partial_reconfig) {
		/* Disable bridge */
		socfpga_bridges_disable();
	}

243
244
245
	return 0;
}

Hadi Asyrafi's avatar
Hadi Asyrafi committed
246
247
248
249
250
251
252
253
static bool is_fpga_config_buffer_full(void)
{
	for (int i = 0; i < FPGA_CONFIG_BUFFER_SIZE; i++)
		if (!fpga_config_buffers[i].write_requested)
			return false;
	return true;
}

254
static bool is_address_in_ddr_range(uint64_t addr, uint64_t size)
Hadi Asyrafi's avatar
Hadi Asyrafi committed
255
{
256
257
	if (size > (UINT64_MAX - addr))
		return false;
258
	if (addr < BL31_LIMIT)
259
260
261
		return false;
	if (addr + size > DRAM_BASE + DRAM_SIZE)
		return false;
Hadi Asyrafi's avatar
Hadi Asyrafi committed
262

263
	return true;
Hadi Asyrafi's avatar
Hadi Asyrafi committed
264
}
265

266
static uint32_t intel_fpga_config_write(uint64_t mem, uint64_t size)
267
{
Hadi Asyrafi's avatar
Hadi Asyrafi committed
268
	int i;
269

Hadi Asyrafi's avatar
Hadi Asyrafi committed
270
	intel_fpga_sdm_write_all();
271

272
	if (!is_address_in_ddr_range(mem, size) ||
Hadi Asyrafi's avatar
Hadi Asyrafi committed
273
274
		is_fpga_config_buffer_full())
		return INTEL_SIP_SMC_STATUS_REJECTED;
275
276

	for (i = 0; i < FPGA_CONFIG_BUFFER_SIZE; i++) {
Hadi Asyrafi's avatar
Hadi Asyrafi committed
277
278
279
280
281
282
283
284
		int j = (i + current_buffer) % FPGA_CONFIG_BUFFER_SIZE;

		if (!fpga_config_buffers[j].write_requested) {
			fpga_config_buffers[j].addr = mem;
			fpga_config_buffers[j].size = size;
			fpga_config_buffers[j].size_written = 0;
			fpga_config_buffers[j].write_requested = 1;
			fpga_config_buffers[j].block_number =
285
				blocks_submitted++;
Hadi Asyrafi's avatar
Hadi Asyrafi committed
286
			fpga_config_buffers[j].subblocks_sent = 0;
287
288
289
290
			break;
		}
	}

Hadi Asyrafi's avatar
Hadi Asyrafi committed
291
292
	if (is_fpga_config_buffer_full())
		return INTEL_SIP_SMC_STATUS_BUSY;
293

Hadi Asyrafi's avatar
Hadi Asyrafi committed
294
	return INTEL_SIP_SMC_STATUS_OK;
295
296
}

297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
static int is_out_of_sec_range(uint64_t reg_addr)
{
	switch (reg_addr) {
	case(0xF8011100):	/* ECCCTRL1 */
	case(0xF8011104):	/* ECCCTRL2 */
	case(0xF8011110):	/* ERRINTEN */
	case(0xF8011114):	/* ERRINTENS */
	case(0xF8011118):	/* ERRINTENR */
	case(0xF801111C):	/* INTMODE */
	case(0xF8011120):	/* INTSTAT */
	case(0xF8011124):	/* DIAGINTTEST */
	case(0xF801112C):	/* DERRADDRA */
	case(0xFFD12028):	/* SDMMCGRP_CTRL */
	case(0xFFD12044):	/* EMAC0 */
	case(0xFFD12048):	/* EMAC1 */
	case(0xFFD1204C):	/* EMAC2 */
	case(0xFFD12090):	/* ECC_INT_MASK_VALUE */
	case(0xFFD12094):	/* ECC_INT_MASK_SET */
	case(0xFFD12098):	/* ECC_INT_MASK_CLEAR */
	case(0xFFD1209C):	/* ECC_INTSTATUS_SERR */
	case(0xFFD120A0):	/* ECC_INTSTATUS_DERR */
	case(0xFFD120C0):	/* NOC_TIMEOUT */
	case(0xFFD120C4):	/* NOC_IDLEREQ_SET */
	case(0xFFD120C8):	/* NOC_IDLEREQ_CLR */
	case(0xFFD120D0):	/* NOC_IDLEACK */
	case(0xFFD120D4):	/* NOC_IDLESTATUS */
	case(0xFFD12200):	/* BOOT_SCRATCH_COLD0 */
	case(0xFFD12204):	/* BOOT_SCRATCH_COLD1 */
	case(0xFFD12220):	/* BOOT_SCRATCH_COLD8 */
	case(0xFFD12224):	/* BOOT_SCRATCH_COLD9 */
		return 0;

	default:
		break;
	}

	return -1;
}

/* Secure register access */
uint32_t intel_secure_reg_read(uint64_t reg_addr, uint32_t *retval)
{
	if (is_out_of_sec_range(reg_addr))
		return INTEL_SIP_SMC_STATUS_ERROR;

	*retval = mmio_read_32(reg_addr);

	return INTEL_SIP_SMC_STATUS_OK;
}

uint32_t intel_secure_reg_write(uint64_t reg_addr, uint32_t val,
				uint32_t *retval)
{
	if (is_out_of_sec_range(reg_addr))
		return INTEL_SIP_SMC_STATUS_ERROR;

	mmio_write_32(reg_addr, val);

	return intel_secure_reg_read(reg_addr, retval);
}

uint32_t intel_secure_reg_update(uint64_t reg_addr, uint32_t mask,
				 uint32_t val, uint32_t *retval)
{
	if (!intel_secure_reg_read(reg_addr, retval)) {
		*retval &= ~mask;
		*retval |= val;
		return intel_secure_reg_write(reg_addr, *retval, retval);
	}

	return INTEL_SIP_SMC_STATUS_ERROR;
}

370
371
372
373
374
375
/* Intel Remote System Update (RSU) services */
uint64_t intel_rsu_update_address;

static uint32_t intel_rsu_status(uint64_t *respbuf, uint32_t respbuf_sz)
{
	if (mailbox_rsu_status((uint32_t *)respbuf, respbuf_sz) < 0)
376
		return INTEL_SIP_SMC_RSU_ERROR;
377
378
379
380
381
382
383
384
385
386

	return INTEL_SIP_SMC_STATUS_OK;
}

static uint32_t intel_rsu_update(uint64_t update_address)
{
	intel_rsu_update_address = update_address;
	return INTEL_SIP_SMC_STATUS_OK;
}

387
static uint32_t intel_rsu_notify(uint32_t execution_stage)
388
{
389
	if (mailbox_hps_stage_notify(execution_stage) < 0)
390
		return INTEL_SIP_SMC_RSU_ERROR;
391
392
393
394
395
396
397
398

	return INTEL_SIP_SMC_STATUS_OK;
}

static uint32_t intel_rsu_retry_counter(uint32_t *respbuf, uint32_t respbuf_sz,
					uint32_t *ret_stat)
{
	if (mailbox_rsu_status((uint32_t *)respbuf, respbuf_sz) < 0)
399
		return INTEL_SIP_SMC_RSU_ERROR;
400
401
402
403
404

	*ret_stat = respbuf[8];
	return INTEL_SIP_SMC_STATUS_OK;
}

405
/* Mailbox services */
406
static uint32_t intel_mbox_send_cmd(uint32_t cmd, uint32_t *args, int len,
407
408
409
410
				    int urgent, uint32_t *response,
				    int resp_len, int *mbox_status,
				    int *len_in_resp)
{
411
412
413
414
415
416
	*len_in_resp = 0;
	*mbox_status = 0;

	if (!is_address_in_ddr_range((uint64_t)args, sizeof(uint32_t) * len))
		return INTEL_SIP_SMC_STATUS_REJECTED;

417
418
419
420
421
422
423
424
425
426
427
428
429
	int status = mailbox_send_cmd(MBOX_JOB_ID, cmd, args, len, urgent,
				      response, resp_len);

	if (status < 0) {
		*mbox_status = -status;
		return INTEL_SIP_SMC_STATUS_ERROR;
	}

	*mbox_status = 0;
	*len_in_resp = status;
	return INTEL_SIP_SMC_STATUS_OK;
}

430
431
432
433
434
435
436
437
438
439
440
441
442
/*
 * This function is responsible for handling all SiP calls from the NS world
 */

uintptr_t sip_smc_handler(uint32_t smc_fid,
			 u_register_t x1,
			 u_register_t x2,
			 u_register_t x3,
			 u_register_t x4,
			 void *cookie,
			 void *handle,
			 u_register_t flags)
{
443
	uint32_t val = 0;
444
445
	uint32_t status = INTEL_SIP_SMC_STATUS_OK;
	uint32_t completed_addr[3];
446
	uint64_t rsu_respbuf[9];
447
	uint32_t count = 0;
448
449
	u_register_t x5, x6;
	int mbox_status, len_in_resp;
450

451

452
453
454
455
	switch (smc_fid) {
	case SIP_SVC_UID:
		/* Return UID to the caller */
		SMC_UUID_RET(handle, intl_svc_uid);
456

457
	case INTEL_SIP_SMC_FPGA_CONFIG_ISDONE:
458
		status = intel_mailbox_fpga_config_isdone(x1);
459
		SMC_RET4(handle, status, 0, 0, 0);
460

461
462
463
464
465
	case INTEL_SIP_SMC_FPGA_CONFIG_GET_MEM:
		SMC_RET3(handle, INTEL_SIP_SMC_STATUS_OK,
			INTEL_SIP_SMC_FPGA_CONFIG_ADDR,
			INTEL_SIP_SMC_FPGA_CONFIG_SIZE -
				INTEL_SIP_SMC_FPGA_CONFIG_ADDR);
466

467
468
469
	case INTEL_SIP_SMC_FPGA_CONFIG_START:
		status = intel_fpga_config_start(x1);
		SMC_RET4(handle, status, 0, 0, 0);
470

471
472
473
	case INTEL_SIP_SMC_FPGA_CONFIG_WRITE:
		status = intel_fpga_config_write(x1, x2);
		SMC_RET4(handle, status, 0, 0, 0);
474

475
476
477
478
479
480
481
	case INTEL_SIP_SMC_FPGA_CONFIG_COMPLETED_WRITE:
		status = intel_fpga_config_completed_write(completed_addr,
								&count);
		switch (count) {
		case 1:
			SMC_RET4(handle, INTEL_SIP_SMC_STATUS_OK,
				completed_addr[0], 0, 0);
482

483
484
485
486
		case 2:
			SMC_RET4(handle, INTEL_SIP_SMC_STATUS_OK,
				completed_addr[0],
				completed_addr[1], 0);
487

488
489
490
491
492
		case 3:
			SMC_RET4(handle, INTEL_SIP_SMC_STATUS_OK,
				completed_addr[0],
				completed_addr[1],
				completed_addr[2]);
493

494
495
		case 0:
			SMC_RET4(handle, status, 0, 0, 0);
496

497
		default:
498
			mailbox_clear_response();
499
500
			SMC_RET1(handle, INTEL_SIP_SMC_STATUS_ERROR);
		}
501
502
503
504
505
506
507
508
509
510
511
512
513

	case INTEL_SIP_SMC_REG_READ:
		status = intel_secure_reg_read(x1, &val);
		SMC_RET3(handle, status, val, x1);

	case INTEL_SIP_SMC_REG_WRITE:
		status = intel_secure_reg_write(x1, (uint32_t)x2, &val);
		SMC_RET3(handle, status, val, x1);

	case INTEL_SIP_SMC_REG_UPDATE:
		status = intel_secure_reg_update(x1, (uint32_t)x2,
						 (uint32_t)x3, &val);
		SMC_RET3(handle, status, val, x1);
514

515
516
517
518
519
520
521
522
523
524
525
526
527
528
529
530
531
532
533
534
535
536
537
538
539
540
541
	case INTEL_SIP_SMC_RSU_STATUS:
		status = intel_rsu_status(rsu_respbuf,
					ARRAY_SIZE(rsu_respbuf));
		if (status) {
			SMC_RET1(handle, status);
		} else {
			SMC_RET4(handle, rsu_respbuf[0], rsu_respbuf[1],
					rsu_respbuf[2], rsu_respbuf[3]);
		}

	case INTEL_SIP_SMC_RSU_UPDATE:
		status = intel_rsu_update(x1);
		SMC_RET1(handle, status);

	case INTEL_SIP_SMC_RSU_NOTIFY:
		status = intel_rsu_notify(x1);
		SMC_RET1(handle, status);

	case INTEL_SIP_SMC_RSU_RETRY_COUNTER:
		status = intel_rsu_retry_counter((uint32_t *)rsu_respbuf,
						ARRAY_SIZE(rsu_respbuf), &val);
		if (status) {
			SMC_RET1(handle, status);
		} else {
			SMC_RET2(handle, status, val);
		}

542
543
544
	case INTEL_SIP_SMC_MBOX_SEND_CMD:
		x5 = SMC_GET_GP(handle, CTX_GPREG_X5);
		x6 = SMC_GET_GP(handle, CTX_GPREG_X6);
545
		status = intel_mbox_send_cmd(x1, (uint32_t *)x2, x3, x4,
546
547
548
549
					     (uint32_t *)x5, x6, &mbox_status,
					     &len_in_resp);
		SMC_RET4(handle, status, mbox_status, x5, len_in_resp);

550
551
552
553
554
555
556
	default:
		return socfpga_sip_handler(smc_fid, x1, x2, x3, x4,
			cookie, handle, flags);
	}
}

DECLARE_RT_SVC(
557
	socfpga_sip_svc,
558
559
560
561
562
563
564
565
	OEN_SIP_START,
	OEN_SIP_END,
	SMC_TYPE_FAST,
	NULL,
	sip_smc_handler
);

DECLARE_RT_SVC(
566
	socfpga_sip_svc_std,
567
568
569
570
571
572
	OEN_SIP_START,
	OEN_SIP_END,
	SMC_TYPE_YIELD,
	NULL,
	sip_smc_handler
);