pm_api_sys.c 31.1 KB
Newer Older
1
/*
2
 * Copyright (c) 2019-2020, Xilinx, Inc. All rights reserved.
3
4
5
6
7
8
9
10
11
12
13
 *
 * SPDX-License-Identifier: BSD-3-Clause
 */

/*
 * Versal system level PM-API functions and communication with PMC via
 * IPI interrupts
 */

#include <pm_common.h>
#include <pm_ipi.h>
14
#include <plat/common/platform.h>
15
16
#include "pm_api_sys.h"
#include "pm_client.h"
17
#include "pm_defs.h"
18
19
20
21
22
23
24

/*********************************************************************
 * Target module IDs macros
 ********************************************************************/
#define LIBPM_MODULE_ID		0x2
#define LOADER_MODULE_ID	0x7

25
26
27
28
29
30
31
32
33
34
35
36
37
/* default shutdown/reboot scope is system(2) */
static unsigned int pm_shutdown_scope = XPM_SHUTDOWN_SUBTYPE_RST_SYSTEM;

/**
 * pm_get_shutdown_scope() - Get the currently set shutdown scope
 *
 * @return	Shutdown scope value
 */
unsigned int pm_get_shutdown_scope(void)
{
	return pm_shutdown_scope;
}

38
39
40
/**
 * Assigning of argument values into array elements.
 */
41
42
#define PM_PACK_PAYLOAD1(pl, mid, flag, arg0) {	\
	pl[0] = (uint32_t)((uint32_t)((arg0) & 0xFF) | (mid << 8) | ((flag) << 24)); \
43
44
}

45
46
47
#define PM_PACK_PAYLOAD2(pl, mid, flag, arg0, arg1) {		\
	pl[1] = (uint32_t)(arg1);				\
	PM_PACK_PAYLOAD1(pl, mid, flag, arg0);			\
48
49
}

50
51
52
#define PM_PACK_PAYLOAD3(pl, mid, flag, arg0, arg1, arg2) {	\
	pl[2] = (uint32_t)(arg2);				\
	PM_PACK_PAYLOAD2(pl, mid, flag, arg0, arg1);		\
53
54
}

55
56
57
#define PM_PACK_PAYLOAD4(pl, mid, flag, arg0, arg1, arg2, arg3) {	\
	pl[3] = (uint32_t)(arg3);					\
	PM_PACK_PAYLOAD3(pl, mid, flag, arg0, arg1, arg2);		\
58
59
}

60
#define PM_PACK_PAYLOAD5(pl, mid, flag, arg0, arg1, arg2, arg3, arg4) {	\
61
	pl[4] = (uint32_t)(arg4);					\
62
	PM_PACK_PAYLOAD4(pl, mid, flag, arg0, arg1, arg2, arg3);	\
63
64
}

65
66
67
#define PM_PACK_PAYLOAD6(pl, mid, flag, arg0, arg1, arg2, arg3, arg4, arg5) {	\
	pl[5] = (uint32_t)(arg5);						\
	PM_PACK_PAYLOAD5(pl, mid, flag, arg0, arg1, arg2, arg3, arg4);		\
68
69
}

70
71
72
73
74
/* PM API functions */

/**
 * pm_get_api_version() - Get version number of PMC PM firmware
 * @version	Returns 32-bit version number of PMC Power Management Firmware
75
76
 * @flag	0 - Call from secure source
 *		1 - Call from non-secure source
77
78
79
 *
 * @return	Returns status, either success or error+reason
 */
80
enum pm_ret_status pm_get_api_version(unsigned int *version, uint32_t flag)
81
82
83
84
{
	uint32_t payload[PAYLOAD_ARG_CNT];

	/* Send request to the PMC */
85
	PM_PACK_PAYLOAD1(payload, LIBPM_MODULE_ID, flag, PM_GET_API_VERSION);
86
87
	return pm_ipi_send_sync(primary_proc, payload, version, 1);
}
88

89
90
91
92
/**
 * pm_init_finalize() - Call to notify PMC PM firmware that master has power
 *			management enabled and that it has finished its
 *			initialization
93
94
 * @flag	0 - Call from secure source
 *		1 - Call from non-secure source
95
96
97
 *
 * @return	Status returned by the PMU firmware
 */
98
enum pm_ret_status pm_init_finalize(uint32_t flag)
99
100
101
102
{
	uint32_t payload[PAYLOAD_ARG_CNT];

	/* Send request to the PMU */
103
	PM_PACK_PAYLOAD1(payload, LIBPM_MODULE_ID, flag, PM_INIT_FINALIZE);
104
105
106
	return pm_ipi_send_sync(primary_proc, payload, NULL, 0);
}

107
108
109
110
111
112
/**
 * pm_self_suspend() - PM call for processor to suspend itself
 * @nid		Node id of the processor or subsystem
 * @latency	Requested maximum wakeup latency (not supported)
 * @state	Requested state
 * @address	Resume address
113
114
 * @flag	0 - Call from secure source
 *		1 - Call from non-secure source
115
116
117
118
119
120
121
122
123
 *
 * This is a blocking call, it will return only once PMU has responded.
 * On a wakeup, resume address will be automatically set by PMU.
 *
 * @return	Returns status, either success or error+reason
 */
enum pm_ret_status pm_self_suspend(uint32_t nid,
				   unsigned int latency,
				   unsigned int state,
124
				   uintptr_t address, uint32_t flag)
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
{
	uint32_t payload[PAYLOAD_ARG_CNT];
	unsigned int cpuid = plat_my_core_pos();
	const struct pm_proc *proc = pm_get_proc(cpuid);

	if (!proc) {
		WARN("Failed to get proc %d\n", cpuid);
		return PM_RET_ERROR_INTERNAL;
	}

	/*
	 * Do client specific suspend operations
	 * (e.g. set powerdown request bit)
	 */
	pm_client_suspend(proc, state);

	/* Send request to the PLM */
142
	PM_PACK_PAYLOAD6(payload, LIBPM_MODULE_ID, flag, PM_SELF_SUSPEND,
143
144
145
146
147
148
149
150
151
			 proc->node_id, latency, state, address,
			 (address >> 32));
	return pm_ipi_send_sync(proc, payload, NULL, 0);
}

/**
 * pm_abort_suspend() - PM call to announce that a prior suspend request
 *			is to be aborted.
 * @reason	Reason for the abort
152
153
 * @flag	0 - Call from secure source
 *		1 - Call from non-secure source
154
155
156
157
158
159
 *
 * Calling PU expects the PMU to abort the initiated suspend procedure.
 * This is a non-blocking call without any acknowledge.
 *
 * @return	Returns status, either success or error+reason
 */
160
enum pm_ret_status pm_abort_suspend(enum pm_abort_reason reason, uint32_t flag)
161
162
163
164
165
166
167
168
169
170
{
	uint32_t payload[PAYLOAD_ARG_CNT];

	/*
	 * Do client specific abort suspend operations
	 * (e.g. enable interrupts and clear powerdown request bit)
	 */
	pm_client_abort_suspend();

	/* Send request to the PLM */
171
172
	PM_PACK_PAYLOAD3(payload, LIBPM_MODULE_ID, flag, PM_ABORT_SUSPEND,
			 reason, primary_proc->node_id);
173
174
175
176
177
178
179
180
181
182
	return pm_ipi_send(primary_proc, payload);
}

/**
 * pm_req_suspend() - PM call to request for another PU or subsystem to
 *		      be suspended gracefully.
 * @target	Node id of the targeted PU or subsystem
 * @ack		Flag to specify whether acknowledge is requested
 * @latency	Requested wakeup latency (not supported)
 * @state	Requested state (not supported)
183
184
 * @flag	0 - Call from secure source
 *		1 - Call from non-secure source
185
186
187
188
 *
 * @return	Returns status, either success or error+reason
 */
enum pm_ret_status pm_req_suspend(uint32_t target, uint8_t ack,
189
190
				  unsigned int latency, unsigned int state,
				  uint32_t flag)
191
192
193
194
{
	uint32_t payload[PAYLOAD_ARG_CNT];

	/* Send request to the PMU */
195
	PM_PACK_PAYLOAD4(payload, LIBPM_MODULE_ID, flag, PM_REQ_SUSPEND, target,
196
197
198
199
200
201
			 latency, state);
	if (ack == IPI_BLOCKING)
		return pm_ipi_send_sync(primary_proc, payload, NULL, 0);
	else
		return pm_ipi_send(primary_proc, payload);
}
202

203
204
205
206
207
208
209
210
/**
 * pm_req_wakeup() - PM call for processor to wake up selected processor
 *		     or subsystem
 * @target	Device ID of the processor or subsystem to wake up
 * @set_address	Resume address presence indicator
 *		1 - resume address specified, 0 - otherwise
 * @address	Resume address
 * @ack		Flag to specify whether acknowledge requested
211
212
 * @flag	0 - Call from secure source
 *		1 - Call from non-secure source
213
214
215
216
217
218
219
220
221
 *
 * This API function is either used to power up another APU core for SMP
 * (by PSCI) or to power up an entirely different PU or subsystem, such
 * as RPU0, RPU, or PL_CORE_xx. Resume address for the target PU will be
 * automatically set by PMC.
 *
 * @return	Returns status, either success or error+reason
 */
enum pm_ret_status pm_req_wakeup(uint32_t target, uint32_t set_address,
222
				 uintptr_t address, uint8_t ack, uint32_t flag)
223
224
225
226
{
	uint32_t payload[PAYLOAD_ARG_CNT];

	/* Send request to the PMC to perform the wake of the PU */
227
	PM_PACK_PAYLOAD5(payload, LIBPM_MODULE_ID, flag, PM_REQ_WAKEUP, target,
228
229
230
231
232
			 set_address, address, ack);

	return pm_ipi_send_sync(primary_proc, payload, NULL, 0);
}

233
234
235
236
237
238
/**
 * pm_request_device() - Request a device
 * @device_id		Device ID
 * @capabilities	Requested capabilities for the device
 * @qos			Required Quality of Service
 * @ack			Flag to specify whether acknowledge requested
239
240
 * @flag		0 - Call from secure source
 *			1 - Call from non-secure source
241
242
243
244
 *
 * @return	Returns status, either success or error+reason
 */
enum pm_ret_status pm_request_device(uint32_t device_id, uint32_t capabilities,
245
				     uint32_t qos, uint32_t ack, uint32_t flag)
246
247
248
249
{
	uint32_t payload[PAYLOAD_ARG_CNT];

	/* Send request to the PMC */
250
	PM_PACK_PAYLOAD5(payload, LIBPM_MODULE_ID, flag, PM_REQUEST_DEVICE,
251
252
253
254
255
256
257
258
			 device_id, capabilities, qos, ack);

	return pm_ipi_send_sync(primary_proc, payload, NULL, 0);
}

/**
 * pm_release_device() - Release a device
 * @device_id		Device ID
259
260
 * @flag		0 - Call from secure source
 *			1 - Call from non-secure source
261
262
263
 *
 * @return	Returns status, either success or error+reason
 */
264
enum pm_ret_status pm_release_device(uint32_t device_id, uint32_t flag)
265
266
267
268
{
	uint32_t payload[PAYLOAD_ARG_CNT];

	/* Send request to the PMC */
269
	PM_PACK_PAYLOAD2(payload, LIBPM_MODULE_ID, flag, PM_RELEASE_DEVICE,
270
271
272
273
274
275
276
277
278
279
280
			 device_id);

	return pm_ipi_send_sync(primary_proc, payload, NULL, 0);
}

/**
 * pm_set_requirement() - Set requirement for the device
 * @device_id		Device ID
 * @capabilities	Requested capabilities for the device
 * @latency		Requested maximum latency
 * @qos			Required Quality of Service
281
282
 * @flag		0 - Call from secure source
 *			1 - Call from non-secure source
283
284
285
286
 *
 * @return	Returns status, either success or error+reason
 */
enum pm_ret_status pm_set_requirement(uint32_t device_id, uint32_t capabilities,
287
288
				      uint32_t latency, uint32_t qos,
				      uint32_t flag)
289
290
291
292
{
	uint32_t payload[PAYLOAD_ARG_CNT];

	/* Send request to the PMC */
293
	PM_PACK_PAYLOAD5(payload, LIBPM_MODULE_ID, flag, PM_SET_REQUIREMENT,
294
295
296
297
298
299
300
301
302
			 device_id, capabilities, latency, qos);

	return pm_ipi_send_sync(primary_proc, payload, NULL, 0);
}

/**
 * pm_get_device_status() - Get device's status
 * @device_id		Device ID
 * @response		Buffer to store device status response
303
304
 * @flag		0 - Call from secure source
 *			1 - Call from non-secure source
305
306
307
 *
 * @return	Returns status, either success or error+reason
 */
308
309
enum pm_ret_status pm_get_device_status(uint32_t device_id, uint32_t *response,
					uint32_t flag)
310
311
312
313
{
	uint32_t payload[PAYLOAD_ARG_CNT];

	/* Send request to the PMC */
314
	PM_PACK_PAYLOAD2(payload, LIBPM_MODULE_ID, flag, PM_GET_DEVICE_STATUS,
315
316
317
318
			 device_id);

	return pm_ipi_send_sync(primary_proc, payload, response, 3);
}
319
320
321
322
323

/**
 * pm_reset_assert() - Assert/De-assert reset
 * @reset	Reset ID
 * @assert	Assert (1) or de-assert (0)
324
325
 * @flag	0 - Call from secure source
 *		1 - Call from non-secure source
326
327
328
 *
 * @return	Returns status, either success or error+reason
 */
329
enum pm_ret_status pm_reset_assert(uint32_t reset, bool assert, uint32_t flag)
330
331
332
333
{
	uint32_t payload[PAYLOAD_ARG_CNT];

	/* Send request to the PMC */
334
	PM_PACK_PAYLOAD3(payload, LIBPM_MODULE_ID, flag, PM_RESET_ASSERT, reset,
335
336
337
338
339
340
341
342
343
			 assert);

	return pm_ipi_send_sync(primary_proc, payload, NULL, 0);
}

/**
 * pm_reset_get_status() - Get current status of a reset line
 * @reset	Reset ID
 * @status	Returns current status of selected reset ID
344
345
 * @flag	0 - Call from secure source
 *		1 - Call from non-secure source
346
347
348
 *
 * @return	Returns status, either success or error+reason
 */
349
350
enum pm_ret_status pm_reset_get_status(uint32_t reset, uint32_t *status,
				       uint32_t flag)
351
352
353
354
{
	uint32_t payload[PAYLOAD_ARG_CNT];

	/* Send request to the PMC */
355
356
	PM_PACK_PAYLOAD2(payload, LIBPM_MODULE_ID, flag, PM_RESET_ASSERT,
			 reset);
357
358
359

	return pm_ipi_send_sync(primary_proc, payload, status, 1);
}
360

361
362
363
/**
 * pm_get_callbackdata() - Read from IPI response buffer
 * @data - array of PAYLOAD_ARG_CNT elements
364
365
 * @flag - 0 - Call from secure source
 *	   1 - Call from non-secure source
366
367
368
 *
 * Read value from ipi buffer response buffer.
 */
369
void pm_get_callbackdata(uint32_t *data, size_t count, uint32_t flag)
370
371
372
373
374
375
376
377
378
{
	/* Return if interrupt is not from PMU */
	if (!pm_ipi_irq_status(primary_proc))
		return;

	pm_ipi_buff_read_callb(data, count);
	pm_ipi_irq_clear(primary_proc);
}

379
380
381
/**
 * pm_pinctrl_request() - Request a pin
 * @pin		Pin ID
382
383
 * @flag	0 - Call from secure source
 *		1 - Call from non-secure source
384
385
386
 *
 * @return	Returns status, either success or error+reason
 */
387
enum pm_ret_status pm_pinctrl_request(uint32_t pin, uint32_t flag)
388
389
390
391
{
	uint32_t payload[PAYLOAD_ARG_CNT];

	/* Send request to the PMC */
392
393
	PM_PACK_PAYLOAD2(payload, LIBPM_MODULE_ID, flag, PM_PINCTRL_REQUEST,
			 pin);
394
395
396
397
398
399
400

	return pm_ipi_send_sync(primary_proc, payload, NULL, 0);
}

/**
 * pm_pinctrl_release() - Release a pin
 * @pin		Pin ID
401
402
 * @flag	0 - Call from secure source
 *		1 - Call from non-secure source
403
404
405
 *
 * @return	Returns status, either success or error+reason
 */
406
enum pm_ret_status pm_pinctrl_release(uint32_t pin, uint32_t flag)
407
408
409
410
{
	uint32_t payload[PAYLOAD_ARG_CNT];

	/* Send request to the PMC */
411
412
	PM_PACK_PAYLOAD2(payload, LIBPM_MODULE_ID, flag, PM_PINCTRL_RELEASE,
			 pin);
413
414
415
416
417
418
419
420

	return pm_ipi_send_sync(primary_proc, payload, NULL, 0);
}

/**
 * pm_pinctrl_set_function() - Set pin function
 * @pin		Pin ID
 * @function	Function ID
421
422
 * @flag	0 - Call from secure source
 *		1 - Call from non-secure source
423
424
425
 *
 * @return	Returns status, either success or error+reason
 */
426
427
enum pm_ret_status pm_pinctrl_set_function(uint32_t pin, uint32_t function,
					   uint32_t flag)
428
429
430
431
{
	uint32_t payload[PAYLOAD_ARG_CNT];

	/* Send request to the PMC */
432
433
	PM_PACK_PAYLOAD3(payload, LIBPM_MODULE_ID, flag,
			 PM_PINCTRL_SET_FUNCTION, pin, function)
434
435
436
437
438
439
440
441

	return pm_ipi_send_sync(primary_proc, payload, NULL, 0);
}

/**
 * pm_pinctrl_get_function() - Get function set on the pin
 * @pin		Pin ID
 * @function	Function set on the pin
442
443
 * @flag	0 - Call from secure source
 *		1 - Call from non-secure source
444
445
446
 *
 * @return	Returns status, either success or error+reason
 */
447
448
enum pm_ret_status pm_pinctrl_get_function(uint32_t pin, uint32_t *function,
					   uint32_t flag)
449
450
451
452
{
	uint32_t payload[PAYLOAD_ARG_CNT];

	/* Send request to the PMC */
453
454
	PM_PACK_PAYLOAD2(payload, LIBPM_MODULE_ID, flag,
			 PM_PINCTRL_SET_FUNCTION, pin);
455
456
457
458
459
460
461
462
463

	return pm_ipi_send_sync(primary_proc, payload, function, 1);
}

/**
 * pm_pinctrl_set_pin_param() - Set configuration parameter for the pin
 * @pin		Pin ID
 * @param	Parameter ID
 * @value	Parameter value
464
465
 * @flag	0 - Call from secure source
 *		1 - Call from non-secure source
466
467
468
469
 *
 * @return	Returns status, either success or error+reason
 */
enum pm_ret_status pm_pinctrl_set_pin_param(uint32_t pin, uint32_t param,
470
					    uint32_t value, uint32_t flag)
471
472
473
474
{
	uint32_t payload[PAYLOAD_ARG_CNT];

	/* Send request to the PMC */
475
476
	PM_PACK_PAYLOAD4(payload, LIBPM_MODULE_ID, flag,
			 PM_PINCTRL_CONFIG_PARAM_SET, pin, param, value);
477
478
479
480
481
482
483
484
485

	return pm_ipi_send_sync(primary_proc, payload, NULL, 0);
}

/**
 * pm_pinctrl_get_pin_param() - Get configuration parameter value for the pin
 * @pin		Pin ID
 * @param	Parameter ID
 * @value	Buffer to store parameter value
486
487
 * @flag	0 - Call from secure source
 *		1 - Call from non-secure source
488
489
490
491
 *
 * @return	Returns status, either success or error+reason
 */
enum pm_ret_status pm_pinctrl_get_pin_param(uint32_t pin, uint32_t param,
492
					    uint32_t *value, uint32_t flag)
493
494
495
496
{
	uint32_t payload[PAYLOAD_ARG_CNT];

	/* Send request to the PMC */
497
498
	PM_PACK_PAYLOAD3(payload, LIBPM_MODULE_ID, flag,
			 PM_PINCTRL_CONFIG_PARAM_GET, pin, param);
499
500
501

	return pm_ipi_send_sync(primary_proc, payload, value, 1);
}
502
503
504
505

/**
 * pm_clock_enable() - Enable the clock
 * @clk_id	Clock ID
506
507
 * @flag	0 - Call from secure source
 *		1 - Call from non-secure source
508
509
510
 *
 * @return	Returns status, either success or error+reason
 */
511
enum pm_ret_status pm_clock_enable(uint32_t clk_id, uint32_t flag)
512
513
514
515
{
	uint32_t payload[PAYLOAD_ARG_CNT];

	/* Send request to the PMC */
516
517
	PM_PACK_PAYLOAD2(payload, LIBPM_MODULE_ID, flag, PM_CLOCK_ENABLE,
			 clk_id);
518
519
520
521
522
523
524

	return pm_ipi_send_sync(primary_proc, payload, NULL, 0);
}

/**
 * pm_clock_disable() - Disable the clock
 * @clk_id	Clock ID
525
526
 * @flag	0 - Call from secure source
 *		1 - Call from non-secure source
527
528
529
 *
 * @return	Returns status, either success or error+reason
 */
530
enum pm_ret_status pm_clock_disable(uint32_t clk_id, uint32_t flag)
531
532
533
534
{
	uint32_t payload[PAYLOAD_ARG_CNT];

	/* Send request to the PMC */
535
536
	PM_PACK_PAYLOAD2(payload, LIBPM_MODULE_ID, flag, PM_CLOCK_DISABLE,
			 clk_id);
537
538
539
540
541
542
543
544

	return pm_ipi_send_sync(primary_proc, payload, NULL, 0);
}

/**
 * pm_clock_get_state() - Get clock status
 * @clk_id	Clock ID
 * @state:	Buffer to store clock status (1: Enabled, 0:Disabled)
545
546
 * @flag	0 - Call from secure source
 *		1 - Call from non-secure source
547
548
549
 *
 * @return	Returns status, either success or error+reason
 */
550
551
enum pm_ret_status pm_clock_get_state(uint32_t clk_id, uint32_t *state,
				      uint32_t flag)
552
553
554
555
{
	uint32_t payload[PAYLOAD_ARG_CNT];

	/* Send request to the PMC */
556
557
	PM_PACK_PAYLOAD2(payload, LIBPM_MODULE_ID, flag, PM_CLOCK_GETSTATE,
			 clk_id);
558
559
560
561
562
563
564
565

	return pm_ipi_send_sync(primary_proc, payload, state, 1);
}

/**
 * pm_clock_set_divider() - Set divider for the clock
 * @clk_id	Clock ID
 * @divider	Divider value
566
567
 * @flag	0 - Call from secure source
 *		1 - Call from non-secure source
568
569
570
 *
 * @return	Returns status, either success or error+reason
 */
571
572
enum pm_ret_status pm_clock_set_divider(uint32_t clk_id, uint32_t divider,
					uint32_t flag)
573
574
575
576
{
	uint32_t payload[PAYLOAD_ARG_CNT];

	/* Send request to the PMC */
577
578
	PM_PACK_PAYLOAD3(payload, LIBPM_MODULE_ID, flag, PM_CLOCK_SETDIVIDER,
			 clk_id, divider);
579
580
581
582
583
584
585
586

	return pm_ipi_send_sync(primary_proc, payload, NULL, 0);
}

/**
 * pm_clock_get_divider() - Get divider value for the clock
 * @clk_id	Clock ID
 * @divider:	Buffer to store clock divider value
587
588
 * @flag	0 - Call from secure source
 *		1 - Call from non-secure source
589
590
591
 *
 * @return	Returns status, either success or error+reason
 */
592
593
enum pm_ret_status pm_clock_get_divider(uint32_t clk_id, uint32_t *divider,
					uint32_t flag)
594
595
596
597
{
	uint32_t payload[PAYLOAD_ARG_CNT];

	/* Send request to the PMC */
598
599
	PM_PACK_PAYLOAD2(payload, LIBPM_MODULE_ID, flag, PM_CLOCK_GETDIVIDER,
			 clk_id);
600
601
602
603
604
605
606
607

	return pm_ipi_send_sync(primary_proc, payload, divider, 1);
}

/**
 * pm_clock_set_parent() - Set parent for the clock
 * @clk_id	Clock ID
 * @parent	Parent ID
608
609
 * @flag	0 - Call from secure source
 *		1 - Call from non-secure source
610
611
612
 *
 * @return	Returns status, either success or error+reason
 */
613
614
enum pm_ret_status pm_clock_set_parent(uint32_t clk_id, uint32_t parent,
				       uint32_t flag)
615
616
617
618
{
	uint32_t payload[PAYLOAD_ARG_CNT];

	/* Send request to the PMC */
619
620
	PM_PACK_PAYLOAD3(payload, LIBPM_MODULE_ID, flag, PM_CLOCK_SETPARENT,
			 clk_id, parent);
621
622
623
624
625
626
627
628

	return pm_ipi_send_sync(primary_proc, payload, NULL, 0);
}

/**
 * pm_clock_get_parent() - Get parent value for the clock
 * @clk_id	Clock ID
 * @parent:	Buffer to store clock parent value
629
630
 * @flag	0 - Call from secure source
 *		1 - Call from non-secure source
631
632
633
 *
 * @return	Returns status, either success or error+reason
 */
634
635
enum pm_ret_status pm_clock_get_parent(uint32_t clk_id, uint32_t *parent,
				       uint32_t flag)
636
637
638
639
{
	uint32_t payload[PAYLOAD_ARG_CNT];

	/* Send request to the PMC */
640
641
	PM_PACK_PAYLOAD2(payload, LIBPM_MODULE_ID, flag, PM_CLOCK_GETPARENT,
			 clk_id);
642
643
644

	return pm_ipi_send_sync(primary_proc, payload, parent, 1);
}
645
646
647
648
/**
 * pm_clock_get_rate() - Get the rate value for the clock
 * @clk_id	Clock ID
 * @rate:	Buffer to store clock rate value
649
650
 * @flag	0 - Call from secure source
 *		1 - Call from non-secure source
651
652
653
 *
 * @return	Returns status, either success or error+reason
 */
654
655
enum pm_ret_status pm_clock_get_rate(uint32_t clk_id, uint32_t *clk_rate,
				     uint32_t flag)
656
657
658
659
{
	uint32_t payload[PAYLOAD_ARG_CNT];

	/* Send request to the PMC */
660
661
	PM_PACK_PAYLOAD2(payload, LIBPM_MODULE_ID, flag, PM_CLOCK_GETRATE,
			 clk_id);
662
663
664

	return pm_ipi_send_sync(primary_proc, payload, clk_rate, 2);
}
665
666
667
668
669
670

/**
 * pm_pll_set_param() - Set PLL parameter
 * @clk_id	PLL clock ID
 * @param	PLL parameter ID
 * @value	Value to set for PLL parameter
671
672
 * @flag	0 - Call from secure source
 *		1 - Call from non-secure source
673
674
675
676
 *
 * @return	Returns status, either success or error+reason
 */
enum pm_ret_status pm_pll_set_param(uint32_t clk_id, uint32_t param,
677
				    uint32_t value, uint32_t flag)
678
679
680
681
{
	uint32_t payload[PAYLOAD_ARG_CNT];

	/* Send request to the PMC */
682
683
	PM_PACK_PAYLOAD4(payload, LIBPM_MODULE_ID, flag, PM_PLL_SET_PARAMETER,
			 clk_id, param, value);
684
685
686
687
688
689
690
691
692

	return pm_ipi_send_sync(primary_proc, payload, NULL, 0);
}

/**
 * pm_pll_get_param() - Get PLL parameter value
 * @clk_id	PLL clock ID
 * @param	PLL parameter ID
 * @value:	Buffer to store PLL parameter value
693
694
 * @flag	0 - Call from secure source
 *		1 - Call from non-secure source
695
696
697
698
 *
 * @return	Returns status, either success or error+reason
 */
enum pm_ret_status pm_pll_get_param(uint32_t clk_id, uint32_t param,
699
				    uint32_t *value, uint32_t flag)
700
701
702
703
{
	uint32_t payload[PAYLOAD_ARG_CNT];

	/* Send request to the PMC */
704
705
	PM_PACK_PAYLOAD3(payload, LIBPM_MODULE_ID, flag, PM_PLL_GET_PARAMETER,
			 clk_id, param);
706
707
708
709
710
711
712
713

	return pm_ipi_send_sync(primary_proc, payload, value, 1);
}

/**
 * pm_pll_set_mode() - Set PLL mode
 * @clk_id	PLL clock ID
 * @mode	PLL mode
714
715
 * @flag	0 - Call from secure source
 *		1 - Call from non-secure source
716
717
718
 *
 * @return	Returns status, either success or error+reason
 */
719
720
enum pm_ret_status pm_pll_set_mode(uint32_t clk_id, uint32_t mode,
				   uint32_t flag)
721
722
723
724
{
	uint32_t payload[PAYLOAD_ARG_CNT];

	/* Send request to the PMC */
725
726
	PM_PACK_PAYLOAD3(payload, LIBPM_MODULE_ID, flag, PM_PLL_SET_MODE,
			 clk_id, mode);
727
728
729
730
731
732
733
734

	return pm_ipi_send_sync(primary_proc, payload, NULL, 0);
}

/**
 * pm_pll_get_mode() - Get PLL mode
 * @clk_id	PLL clock ID
 * @mode:	Buffer to store PLL mode
735
736
 * @flag	0 - Call from secure source
 *		1 - Call from non-secure source
737
738
739
 *
 * @return	Returns status, either success or error+reason
 */
740
741
enum pm_ret_status pm_pll_get_mode(uint32_t clk_id, uint32_t *mode,
				   uint32_t flag)
742
743
744
745
{
	uint32_t payload[PAYLOAD_ARG_CNT];

	/* Send request to the PMC */
746
747
	PM_PACK_PAYLOAD2(payload, LIBPM_MODULE_ID, flag, PM_PLL_GET_MODE,
			 clk_id);
748
749
750

	return pm_ipi_send_sync(primary_proc, payload, mode, 1);
}
751
752
753
754
755
756

/**
 * pm_force_powerdown() - PM call to request for another PU or subsystem to
 *			  be powered down forcefully
 * @target	Device ID of the PU node to be forced powered down.
 * @ack		Flag to specify whether acknowledge is requested
757
758
 * @flag	0 - Call from secure source
 *		1 - Call from non-secure source
759
760
761
 *
 * @return	Returns status, either success or error+reason
 */
762
763
enum pm_ret_status pm_force_powerdown(uint32_t target, uint8_t ack,
				      uint32_t flag)
764
765
766
767
{
	uint32_t payload[PAYLOAD_ARG_CNT];

	/* Send request to the PMC */
768
769
	PM_PACK_PAYLOAD3(payload, LIBPM_MODULE_ID, flag, PM_FORCE_POWERDOWN,
			 target, ack);
770
771
772
773
774
775
776
777
778

	if (ack == IPI_BLOCKING)
		return pm_ipi_send_sync(primary_proc, payload, NULL, 0);
	else
		return pm_ipi_send(primary_proc, payload);
}

/**
 * pm_system_shutdown() - PM call to request a system shutdown or restart
779
 * @type	Shutdown or restart? 0=shutdown, 1=restart, 2=setscope
780
 * @subtype	Scope: 0=APU-subsystem, 1=PS, 2=system
781
782
 * @flag	0 - Call from secure source
 *		1 - Call from non-secure source
783
784
785
 *
 * @return	Returns status, either success or error+reason
 */
786
787
enum pm_ret_status pm_system_shutdown(uint32_t type, uint32_t subtype,
				      uint32_t flag)
788
789
790
{
	uint32_t payload[PAYLOAD_ARG_CNT];

791
792
793
794
795
796
	if (type == XPM_SHUTDOWN_TYPE_SETSCOPE_ONLY) {
		/* Setting scope for subsequent PSCI reboot or shutdown */
		pm_shutdown_scope = subtype;
		return PM_RET_SUCCESS;
	}

797
	/* Send request to the PMC */
798
799
	PM_PACK_PAYLOAD3(payload, LIBPM_MODULE_ID, flag, PM_SYSTEM_SHUTDOWN,
			 type, subtype);
800
801
802

	return pm_ipi_send_non_blocking(primary_proc, payload);
}
803

804
805
806
807
808
809
810
/**
* pm_query_data() -  PM API for querying firmware data
* @qid	The type of data to query
* @arg1	Argument 1 to requested query data call
* @arg2	Argument 2 to requested query data call
* @arg3	Argument 3 to requested query data call
* @data	Returned output data
811
812
* @flag 0 - Call from secure source
*	1 - Call from non-secure source
813
814
815
816
*
* This function returns requested data.
*/
enum pm_ret_status pm_query_data(uint32_t qid, uint32_t arg1, uint32_t arg2,
817
				 uint32_t arg3, uint32_t *data, uint32_t flag)
818
{
819
820
	uint32_t ret;
	uint32_t version;
821
	uint32_t payload[PAYLOAD_ARG_CNT];
822
	uint32_t fw_api_version;
823
824

	/* Send request to the PMC */
825
826
	PM_PACK_PAYLOAD5(payload, LIBPM_MODULE_ID, flag, PM_QUERY_DATA, qid,
			 arg1, arg2, arg3);
827

828
	ret = pm_feature_check(PM_QUERY_DATA, &version, flag);
829
	if (PM_RET_SUCCESS == ret) {
830
831
832
833
834
835
836
837
838
839
840
841
842
843
		fw_api_version = version & 0xFFFF ;
		if ((2U == fw_api_version) &&
		    ((XPM_QID_CLOCK_GET_NAME == qid) ||
		     (XPM_QID_PINCTRL_GET_FUNCTION_NAME == qid))) {
			ret = pm_ipi_send_sync(primary_proc, payload, data, 8);
			ret = data[0];
			data[0] = data[1];
			data[1] = data[2];
			data[2] = data[3];
		} else {
			ret = pm_ipi_send_sync(primary_proc, payload, data, 4);
		}
	}
	return ret;
844
}
845
846
847
848
849
850
851
/**
 * pm_api_ioctl() -  PM IOCTL API for device control and configs
 * @device_id	Device ID
 * @ioctl_id	ID of the requested IOCTL
 * @arg1	Argument 1 to requested IOCTL call
 * @arg2	Argument 2 to requested IOCTL call
 * @value	Returned output value
852
853
 * @flag	0 - Call from secure source
 *		1 - Call from non-secure source
854
855
856
857
858
859
 *
 * This function calls IOCTL to firmware for device control and configuration.
 *
 * @return	Returns status, either success or error+reason
 */
enum pm_ret_status pm_api_ioctl(uint32_t device_id, uint32_t ioctl_id,
860
861
				uint32_t arg1, uint32_t arg2, uint32_t *value,
				uint32_t flag)
862
863
864
865
866
{
	uint32_t payload[PAYLOAD_ARG_CNT];

	switch (ioctl_id) {
	case IOCTL_SET_PLL_FRAC_MODE:
867
		return pm_pll_set_mode(arg1, arg2, flag);
868
	case IOCTL_GET_PLL_FRAC_MODE:
869
		return pm_pll_get_mode(arg1, value, flag);
870
	case IOCTL_SET_PLL_FRAC_DATA:
871
		return pm_pll_set_param(arg1, PM_PLL_PARAM_DATA, arg2, flag);
872
	case IOCTL_GET_PLL_FRAC_DATA:
873
		return pm_pll_get_param(arg1, PM_PLL_PARAM_DATA, value, flag);
874
875
	default:
		/* Send request to the PMC */
876
877
		PM_PACK_PAYLOAD5(payload, LIBPM_MODULE_ID, flag, PM_IOCTL,
				 device_id, ioctl_id, arg1, arg2);
878
879
880
		return pm_ipi_send_sync(primary_proc, payload, value, 1);
	}
}
881
882
883
884
885
886

/**
 * pm_set_wakeup_source() - PM call to specify the wakeup source while suspended
 * @target	Device id of the targeted PU or subsystem
 * @wkup_node	Device id of the wakeup peripheral
 * @enable	Enable or disable the specified peripheral as wake source
887
888
 * @flag	0 - Call from secure source
 *		1 - Call from non-secure source
889
890
891
892
 *
 * @return	Returns status, either success or error+reason
 */
enum pm_ret_status pm_set_wakeup_source(uint32_t target, uint32_t wkup_device,
893
					uint8_t enable, uint32_t flag)
894
895
896
{
	uint32_t payload[PAYLOAD_ARG_CNT];

897
898
	PM_PACK_PAYLOAD4(payload, LIBPM_MODULE_ID, flag, PM_SET_WAKEUP_SOURCE,
			 target, wkup_device, enable);
899
900
	return pm_ipi_send(primary_proc, payload);
}
901

Ravi Patel's avatar
Ravi Patel committed
902
903
904
905
/**
 * pm_get_chipid() - Read silicon ID registers
 * @value       Buffer for return values. Must be large enough
 *		to hold 8 bytes.
906
907
 * @flag	0 - Call from secure source
 *		1 - Call from non-secure source
Ravi Patel's avatar
Ravi Patel committed
908
909
910
 *
 * @return      Returns silicon ID registers
 */
911
enum pm_ret_status pm_get_chipid(uint32_t *value, uint32_t flag)
Ravi Patel's avatar
Ravi Patel committed
912
913
914
915
{
	uint32_t payload[PAYLOAD_ARG_CNT];

	/* Send request to the PMC */
916
	PM_PACK_PAYLOAD1(payload, LIBPM_MODULE_ID, flag, PM_GET_CHIPID);
Ravi Patel's avatar
Ravi Patel committed
917
918
919
920

	return pm_ipi_send_sync(primary_proc, payload, value, 2);
}

921
922
923
924
/**
 * pm_feature_check() - Returns the supported API version if supported
 * @api_id	API ID to check
 * @value	Returned supported API version
925
926
 * @flag	0 - Call from secure source
 *		1 - Call from non-secure source
927
928
929
 *
 * @return	Returns status, either success or error+reason
 */
930
931
enum pm_ret_status pm_feature_check(uint32_t api_id, unsigned int *version,
				    uint32_t flag)
932
933
934
935
936
937
938
{
	uint32_t payload[PAYLOAD_ARG_CNT], fw_api_version;
	uint32_t status;

	switch (api_id) {
	case PM_GET_CALLBACK_DATA:
	case PM_GET_TRUSTZONE_VERSION:
939
	case PM_LOAD_PDI:
940
941
942
943
		*version = (PM_API_BASE_VERSION << 16);
		return PM_RET_SUCCESS;
	case PM_GET_API_VERSION:
	case PM_GET_DEVICE_STATUS:
944
	case PM_GET_OP_CHARACTERISTIC:
945
946
947
948
949
950
951
952
953
954
955
956
	case PM_REQ_SUSPEND:
	case PM_SELF_SUSPEND:
	case PM_FORCE_POWERDOWN:
	case PM_ABORT_SUSPEND:
	case PM_REQ_WAKEUP:
	case PM_SET_WAKEUP_SOURCE:
	case PM_SYSTEM_SHUTDOWN:
	case PM_REQUEST_DEVICE:
	case PM_RELEASE_DEVICE:
	case PM_SET_REQUIREMENT:
	case PM_RESET_ASSERT:
	case PM_RESET_GET_STATUS:
957
	case PM_GET_CHIPID:
958
959
960
961
962
963
964
965
966
967
968
969
970
971
	case PM_PINCTRL_REQUEST:
	case PM_PINCTRL_RELEASE:
	case PM_PINCTRL_GET_FUNCTION:
	case PM_PINCTRL_SET_FUNCTION:
	case PM_PINCTRL_CONFIG_PARAM_GET:
	case PM_PINCTRL_CONFIG_PARAM_SET:
	case PM_IOCTL:
	case PM_CLOCK_ENABLE:
	case PM_CLOCK_DISABLE:
	case PM_CLOCK_GETSTATE:
	case PM_CLOCK_SETDIVIDER:
	case PM_CLOCK_GETDIVIDER:
	case PM_CLOCK_SETPARENT:
	case PM_CLOCK_GETPARENT:
972
	case PM_CLOCK_GETRATE:
973
974
975
976
977
	case PM_PLL_SET_PARAMETER:
	case PM_PLL_GET_PARAMETER:
	case PM_PLL_SET_MODE:
	case PM_PLL_GET_MODE:
	case PM_FEATURE_CHECK:
978
	case PM_INIT_FINALIZE:
979
	case PM_SET_MAX_LATENCY:
980
	case PM_REGISTER_NOTIFIER:
981
982
		*version = (PM_API_BASE_VERSION << 16);
		break;
983
984
985
	case PM_QUERY_DATA:
		*version = (PM_API_QUERY_DATA_VERSION << 16);
		break;
986
987
988
989
990
	default:
		*version = 0U;
		return PM_RET_ERROR_NOFEATURE;
	}

991
992
	PM_PACK_PAYLOAD2(payload, LIBPM_MODULE_ID, flag,
			 PM_FEATURE_CHECK, api_id);
993
994
995
996
997
998
999
1000
1001

	status = pm_ipi_send_sync(primary_proc, payload, &fw_api_version, 1);
	if (status != PM_RET_SUCCESS)
		return status;

	*version |= fw_api_version;

	return PM_RET_SUCCESS;
}
1002
1003
1004
1005
1006
1007
1008
1009
1010

/**
 * pm_load_pdi() - Load the PDI
 *
 * This function provides support to load PDI from linux
 *
 * src:        Source device of pdi(DDR, OCM, SD etc)
 * address_low: lower 32-bit Linear memory space address
 * address_high: higher 32-bit Linear memory space address
1011
1012
 * @flag	0 - Call from secure source
 *		1 - Call from non-secure source
1013
1014
1015
 *
 * @return      Returns status, either success or error+reason
 */
1016
1017
enum pm_ret_status pm_load_pdi(uint32_t src, uint32_t address_low,
			       uint32_t address_high, uint32_t flag)
1018
1019
1020
1021
{
	uint32_t payload[PAYLOAD_ARG_CNT];

	/* Send request to the PMU */
1022
	PM_PACK_PAYLOAD4(payload, LOADER_MODULE_ID, flag, PM_LOAD_PDI, src,
1023
1024
1025
			 address_high, address_low);
	return pm_ipi_send_sync(primary_proc, payload, NULL, 0);
}
1026
1027
1028
1029
1030
1031
1032
1033
1034

/**
 * pm_get_op_characteristic() - PM call to request operating characteristics
 *                              of a device
 * @device_id   Device id
 * @type        Type of the operating characteristic
 *              (power, temperature and latency)
 * @result      Returns the operating characteristic for the requested device,
 *              specified by the type
1035
1036
 * @flag	0 - Call from secure source
 *		1 - Call from non-secure source
1037
1038
1039
1040
1041
 *
 * @return      Returns status, either success or error+reason
 */
enum pm_ret_status pm_get_op_characteristic(uint32_t device_id,
					    enum pm_opchar_type type,
1042
					    uint32_t *result, uint32_t flag)
1043
1044
1045
1046
{
	uint32_t payload[PAYLOAD_ARG_CNT];

	/* Send request to the PMC */
1047
1048
	PM_PACK_PAYLOAD3(payload, LIBPM_MODULE_ID, flag,
			 PM_GET_OP_CHARACTERISTIC, device_id, type);
1049
1050
	return pm_ipi_send_sync(primary_proc, payload, result, 1);
}
1051
1052
1053
1054
1055
1056
1057

/**
 * pm_set_max_latency() - PM call to change in the maximum wake-up latency
 *			  requirements for a specific device currently
 *			  used by that CPU.
 * @device_id	Device ID
 * @latency	Latency value
1058
1059
 * @flag	0 - Call from secure source
 *		1 - Call from non-secure source
1060
1061
1062
 *
 * @return	Returns status, either success or error+reason
 */
1063
1064
enum pm_ret_status pm_set_max_latency(uint32_t device_id, uint32_t latency,
				      uint32_t flag)
1065
1066
1067
1068
{
	uint32_t payload[PAYLOAD_ARG_CNT];

	/* Send request to the PMC */
1069
	PM_PACK_PAYLOAD3(payload, LIBPM_MODULE_ID, flag, PM_SET_MAX_LATENCY,
1070
1071
1072
1073
			 device_id, latency);

	return pm_ipi_send_sync(primary_proc, payload, NULL, 0);
}
1074
1075
1076
1077
1078
1079
1080
1081

/**
 * pm_register_notifier() - PM call to register a subsystem to be notified
 * 			    about the device event
 * @device_id	Device ID for the Node to which the event is related
 * @event	Event in question
 * @wake	Wake subsystem upon capturing the event if value 1
 * @enable	Enable the registration for value 1, disable for value 0
1082
1083
 * @flag	0 - Call from secure source
 *		1 - Call from non-secure source
1084
1085
1086
1087
 *
 * @return	Returns status, either success or error+reason
 */
enum pm_ret_status pm_register_notifier(uint32_t device_id, uint32_t event,
1088
1089
					uint32_t wake, uint32_t enable,
					uint32_t flag)
1090
1091
1092
1093
{
	uint32_t payload[PAYLOAD_ARG_CNT];

	/* Send request to the PMC */
1094
	PM_PACK_PAYLOAD5(payload, LIBPM_MODULE_ID, flag, PM_REGISTER_NOTIFIER,
1095
1096
1097
1098
			 device_id, event, wake, enable);

	return pm_ipi_send_sync(primary_proc, payload, NULL, 0);
}