mirror of
https://github.com/ARM-software/arm-trusted-firmware.git
synced 2025-04-22 04:24:19 +00:00
Merge pull request #834 from douglas-raillard-arm/dr/use_dc_zva_zeroing
Use DC ZVA instruction to zero memory
This commit is contained in:
commit
108e4df7f1
36 changed files with 600 additions and 138 deletions
|
@ -1,5 +1,5 @@
|
||||||
/*
|
/*
|
||||||
* Copyright (c) 2013-2016, ARM Limited and Contributors. All rights reserved.
|
* Copyright (c) 2013-2017, ARM Limited and Contributors. All rights reserved.
|
||||||
*
|
*
|
||||||
* Redistribution and use in source and binary forms, with or without
|
* Redistribution and use in source and binary forms, with or without
|
||||||
* modification, are permitted provided that the following conditions are met:
|
* modification, are permitted provided that the following conditions are met:
|
||||||
|
@ -133,7 +133,8 @@ SECTIONS
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* The .bss section gets initialised to 0 at runtime.
|
* The .bss section gets initialised to 0 at runtime.
|
||||||
* Its base address must be 16-byte aligned.
|
* Its base address should be 16-byte aligned for better performance of the
|
||||||
|
* zero-initialization code.
|
||||||
*/
|
*/
|
||||||
.bss : ALIGN(16) {
|
.bss : ALIGN(16) {
|
||||||
__BSS_START__ = .;
|
__BSS_START__ = .;
|
||||||
|
|
|
@ -1,5 +1,5 @@
|
||||||
/*
|
/*
|
||||||
* Copyright (c) 2015-2016, ARM Limited and Contributors. All rights reserved.
|
* Copyright (c) 2015-2017, ARM Limited and Contributors. All rights reserved.
|
||||||
*
|
*
|
||||||
* Redistribution and use in source and binary forms, with or without
|
* Redistribution and use in source and binary forms, with or without
|
||||||
* modification, are permitted provided that the following conditions are met:
|
* modification, are permitted provided that the following conditions are met:
|
||||||
|
@ -335,7 +335,7 @@ static int bl1_fwu_image_auth(unsigned int image_id,
|
||||||
*/
|
*/
|
||||||
if (image_desc->state == IMAGE_STATE_COPIED) {
|
if (image_desc->state == IMAGE_STATE_COPIED) {
|
||||||
/* Clear the memory.*/
|
/* Clear the memory.*/
|
||||||
memset((void *)base_addr, 0, total_size);
|
zero_normalmem((void *)base_addr, total_size);
|
||||||
flush_dcache_range(base_addr, total_size);
|
flush_dcache_range(base_addr, total_size);
|
||||||
|
|
||||||
/* Indicate that image can be copied again*/
|
/* Indicate that image can be copied again*/
|
||||||
|
|
|
@ -1,5 +1,5 @@
|
||||||
/*
|
/*
|
||||||
* Copyright (c) 2013-2016, ARM Limited and Contributors. All rights reserved.
|
* Copyright (c) 2013-2017, ARM Limited and Contributors. All rights reserved.
|
||||||
*
|
*
|
||||||
* Redistribution and use in source and binary forms, with or without
|
* Redistribution and use in source and binary forms, with or without
|
||||||
* modification, are permitted provided that the following conditions are met:
|
* modification, are permitted provided that the following conditions are met:
|
||||||
|
@ -94,12 +94,12 @@ func bl2_entrypoint
|
||||||
*/
|
*/
|
||||||
ldr x0, =__BSS_START__
|
ldr x0, =__BSS_START__
|
||||||
ldr x1, =__BSS_SIZE__
|
ldr x1, =__BSS_SIZE__
|
||||||
bl zeromem16
|
bl zeromem
|
||||||
|
|
||||||
#if USE_COHERENT_MEM
|
#if USE_COHERENT_MEM
|
||||||
ldr x0, =__COHERENT_RAM_START__
|
ldr x0, =__COHERENT_RAM_START__
|
||||||
ldr x1, =__COHERENT_RAM_UNALIGNED_SIZE__
|
ldr x1, =__COHERENT_RAM_UNALIGNED_SIZE__
|
||||||
bl zeromem16
|
bl zeromem
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
/* --------------------------------------------
|
/* --------------------------------------------
|
||||||
|
|
|
@ -1,5 +1,5 @@
|
||||||
/*
|
/*
|
||||||
* Copyright (c) 2013-2014, ARM Limited and Contributors. All rights reserved.
|
* Copyright (c) 2013-2017, ARM Limited and Contributors. All rights reserved.
|
||||||
*
|
*
|
||||||
* Redistribution and use in source and binary forms, with or without
|
* Redistribution and use in source and binary forms, with or without
|
||||||
* modification, are permitted provided that the following conditions are met:
|
* modification, are permitted provided that the following conditions are met:
|
||||||
|
@ -113,7 +113,8 @@ SECTIONS
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* The .bss section gets initialised to 0 at runtime.
|
* The .bss section gets initialised to 0 at runtime.
|
||||||
* Its base address must be 16-byte aligned.
|
* Its base address should be 16-byte aligned for better performance of the
|
||||||
|
* zero-initialization code.
|
||||||
*/
|
*/
|
||||||
.bss : ALIGN(16) {
|
.bss : ALIGN(16) {
|
||||||
__BSS_START__ = .;
|
__BSS_START__ = .;
|
||||||
|
|
|
@ -1,5 +1,5 @@
|
||||||
/*
|
/*
|
||||||
* Copyright (c) 2015-2016, ARM Limited and Contributors. All rights reserved.
|
* Copyright (c) 2015-2017, ARM Limited and Contributors. All rights reserved.
|
||||||
*
|
*
|
||||||
* Redistribution and use in source and binary forms, with or without
|
* Redistribution and use in source and binary forms, with or without
|
||||||
* modification, are permitted provided that the following conditions are met:
|
* modification, are permitted provided that the following conditions are met:
|
||||||
|
@ -94,7 +94,7 @@ func bl2u_entrypoint
|
||||||
*/
|
*/
|
||||||
ldr x0, =__BSS_START__
|
ldr x0, =__BSS_START__
|
||||||
ldr x1, =__BSS_SIZE__
|
ldr x1, =__BSS_SIZE__
|
||||||
bl zeromem16
|
bl zeromem
|
||||||
|
|
||||||
/* --------------------------------------------
|
/* --------------------------------------------
|
||||||
* Allocate a stack whose memory will be marked
|
* Allocate a stack whose memory will be marked
|
||||||
|
|
|
@ -1,5 +1,5 @@
|
||||||
/*
|
/*
|
||||||
* Copyright (c) 2015, ARM Limited and Contributors. All rights reserved.
|
* Copyright (c) 2015-2017, ARM Limited and Contributors. All rights reserved.
|
||||||
*
|
*
|
||||||
* Redistribution and use in source and binary forms, with or without
|
* Redistribution and use in source and binary forms, with or without
|
||||||
* modification, are permitted provided that the following conditions are met:
|
* modification, are permitted provided that the following conditions are met:
|
||||||
|
@ -100,7 +100,8 @@ SECTIONS
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* The .bss section gets initialised to 0 at runtime.
|
* The .bss section gets initialised to 0 at runtime.
|
||||||
* Its base address must be 16-byte aligned.
|
* Its base address should be 16-byte aligned for better performance of the
|
||||||
|
* zero-initialization code.
|
||||||
*/
|
*/
|
||||||
.bss : ALIGN(16) {
|
.bss : ALIGN(16) {
|
||||||
__BSS_START__ = .;
|
__BSS_START__ = .;
|
||||||
|
|
|
@ -1,5 +1,5 @@
|
||||||
/*
|
/*
|
||||||
* Copyright (c) 2013-2016, ARM Limited and Contributors. All rights reserved.
|
* Copyright (c) 2013-2017, ARM Limited and Contributors. All rights reserved.
|
||||||
*
|
*
|
||||||
* Redistribution and use in source and binary forms, with or without
|
* Redistribution and use in source and binary forms, with or without
|
||||||
* modification, are permitted provided that the following conditions are met:
|
* modification, are permitted provided that the following conditions are met:
|
||||||
|
@ -158,7 +158,8 @@ SECTIONS
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* The .bss section gets initialised to 0 at runtime.
|
* The .bss section gets initialised to 0 at runtime.
|
||||||
* Its base address must be 16-byte aligned.
|
* Its base address should be 16-byte aligned for better performance of the
|
||||||
|
* zero-initialization code.
|
||||||
*/
|
*/
|
||||||
.bss (NOLOAD) : ALIGN(16) {
|
.bss (NOLOAD) : ALIGN(16) {
|
||||||
__BSS_START__ = .;
|
__BSS_START__ = .;
|
||||||
|
|
|
@ -1,5 +1,5 @@
|
||||||
/*
|
/*
|
||||||
* Copyright (c) 2016, ARM Limited and Contributors. All rights reserved.
|
* Copyright (c) 2016-2017, ARM Limited and Contributors. All rights reserved.
|
||||||
*
|
*
|
||||||
* Redistribution and use in source and binary forms, with or without
|
* Redistribution and use in source and binary forms, with or without
|
||||||
* modification, are permitted provided that the following conditions are met:
|
* modification, are permitted provided that the following conditions are met:
|
||||||
|
@ -134,9 +134,10 @@ SECTIONS
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* The .bss section gets initialised to 0 at runtime.
|
* The .bss section gets initialised to 0 at runtime.
|
||||||
* Its base address must be 16-byte aligned.
|
* Its base address should be 8-byte aligned for better performance of the
|
||||||
|
* zero-initialization code.
|
||||||
*/
|
*/
|
||||||
.bss (NOLOAD) : ALIGN(16) {
|
.bss (NOLOAD) : ALIGN(8) {
|
||||||
__BSS_START__ = .;
|
__BSS_START__ = .;
|
||||||
*(.bss*)
|
*(.bss*)
|
||||||
*(COMMON)
|
*(COMMON)
|
||||||
|
|
|
@ -1,5 +1,5 @@
|
||||||
/*
|
/*
|
||||||
* Copyright (c) 2016, ARM Limited and Contributors. All rights reserved.
|
* Copyright (c) 2016-2017, ARM Limited and Contributors. All rights reserved.
|
||||||
*
|
*
|
||||||
* Redistribution and use in source and binary forms, with or without
|
* Redistribution and use in source and binary forms, with or without
|
||||||
* modification, are permitted provided that the following conditions are met:
|
* modification, are permitted provided that the following conditions are met:
|
||||||
|
@ -45,6 +45,7 @@
|
||||||
#include <stdint.h>
|
#include <stdint.h>
|
||||||
#include <string.h>
|
#include <string.h>
|
||||||
#include <types.h>
|
#include <types.h>
|
||||||
|
#include <utils.h>
|
||||||
#include "sp_min_private.h"
|
#include "sp_min_private.h"
|
||||||
|
|
||||||
/* Pointers to per-core cpu contexts */
|
/* Pointers to per-core cpu contexts */
|
||||||
|
@ -203,7 +204,7 @@ void sp_min_warm_boot(void)
|
||||||
smc_set_next_ctx(NON_SECURE);
|
smc_set_next_ctx(NON_SECURE);
|
||||||
|
|
||||||
next_smc_ctx = smc_get_next_ctx();
|
next_smc_ctx = smc_get_next_ctx();
|
||||||
memset(next_smc_ctx, 0, sizeof(smc_ctx_t));
|
zeromem(next_smc_ctx, sizeof(smc_ctx_t));
|
||||||
|
|
||||||
copy_cpu_ctx_to_smc_stx(get_regs_ctx(cm_get_context(NON_SECURE)),
|
copy_cpu_ctx_to_smc_stx(get_regs_ctx(cm_get_context(NON_SECURE)),
|
||||||
next_smc_ctx);
|
next_smc_ctx);
|
||||||
|
|
|
@ -1,5 +1,5 @@
|
||||||
/*
|
/*
|
||||||
* Copyright (c) 2013-2016, ARM Limited and Contributors. All rights reserved.
|
* Copyright (c) 2013-2017, ARM Limited and Contributors. All rights reserved.
|
||||||
*
|
*
|
||||||
* Redistribution and use in source and binary forms, with or without
|
* Redistribution and use in source and binary forms, with or without
|
||||||
* modification, are permitted provided that the following conditions are met:
|
* modification, are permitted provided that the following conditions are met:
|
||||||
|
@ -120,12 +120,12 @@ func tsp_entrypoint
|
||||||
*/
|
*/
|
||||||
ldr x0, =__BSS_START__
|
ldr x0, =__BSS_START__
|
||||||
ldr x1, =__BSS_SIZE__
|
ldr x1, =__BSS_SIZE__
|
||||||
bl zeromem16
|
bl zeromem
|
||||||
|
|
||||||
#if USE_COHERENT_MEM
|
#if USE_COHERENT_MEM
|
||||||
ldr x0, =__COHERENT_RAM_START__
|
ldr x0, =__COHERENT_RAM_START__
|
||||||
ldr x1, =__COHERENT_RAM_UNALIGNED_SIZE__
|
ldr x1, =__COHERENT_RAM_UNALIGNED_SIZE__
|
||||||
bl zeromem16
|
bl zeromem
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
/* --------------------------------------------
|
/* --------------------------------------------
|
||||||
|
|
|
@ -1,5 +1,5 @@
|
||||||
/*
|
/*
|
||||||
* Copyright (c) 2013-2014, ARM Limited and Contributors. All rights reserved.
|
* Copyright (c) 2013-2017, ARM Limited and Contributors. All rights reserved.
|
||||||
*
|
*
|
||||||
* Redistribution and use in source and binary forms, with or without
|
* Redistribution and use in source and binary forms, with or without
|
||||||
* modification, are permitted provided that the following conditions are met:
|
* modification, are permitted provided that the following conditions are met:
|
||||||
|
@ -104,7 +104,8 @@ SECTIONS
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* The .bss section gets initialised to 0 at runtime.
|
* The .bss section gets initialised to 0 at runtime.
|
||||||
* Its base address must be 16-byte aligned.
|
* Its base address should be 16-byte aligned for better performance of the
|
||||||
|
* zero-initialization code.
|
||||||
*/
|
*/
|
||||||
.bss : ALIGN(16) {
|
.bss : ALIGN(16) {
|
||||||
__BSS_START__ = .;
|
__BSS_START__ = .;
|
||||||
|
|
|
@ -1,5 +1,5 @@
|
||||||
/*
|
/*
|
||||||
* Copyright (c) 2013-2016, ARM Limited and Contributors. All rights reserved.
|
* Copyright (c) 2013-2017, ARM Limited and Contributors. All rights reserved.
|
||||||
*
|
*
|
||||||
* Redistribution and use in source and binary forms, with or without
|
* Redistribution and use in source and binary forms, with or without
|
||||||
* modification, are permitted provided that the following conditions are met:
|
* modification, are permitted provided that the following conditions are met:
|
||||||
|
@ -347,7 +347,7 @@ static int load_auth_image_internal(unsigned int image_id,
|
||||||
image_data->image_size);
|
image_data->image_size);
|
||||||
if (rc != 0) {
|
if (rc != 0) {
|
||||||
/* Authentication error, zero memory and flush it right away. */
|
/* Authentication error, zero memory and flush it right away. */
|
||||||
memset((void *)image_data->image_base, 0x00,
|
zero_normalmem((void *)image_data->image_base,
|
||||||
image_data->image_size);
|
image_data->image_size);
|
||||||
flush_dcache_range(image_data->image_base,
|
flush_dcache_range(image_data->image_base,
|
||||||
image_data->image_size);
|
image_data->image_size);
|
||||||
|
@ -543,7 +543,7 @@ static int load_auth_image_internal(meminfo_t *mem_layout,
|
||||||
image_data->image_size);
|
image_data->image_size);
|
||||||
if (rc != 0) {
|
if (rc != 0) {
|
||||||
/* Authentication error, zero memory and flush it right away. */
|
/* Authentication error, zero memory and flush it right away. */
|
||||||
memset((void *)image_data->image_base, 0x00,
|
zero_normalmem((void *)image_data->image_base,
|
||||||
image_data->image_size);
|
image_data->image_size);
|
||||||
flush_dcache_range(image_data->image_base,
|
flush_dcache_range(image_data->image_base,
|
||||||
image_data->image_size);
|
image_data->image_size);
|
||||||
|
|
|
@ -1342,7 +1342,7 @@ All BL images share the following requirements:
|
||||||
|
|
||||||
The following linker symbols are defined for this purpose:
|
The following linker symbols are defined for this purpose:
|
||||||
|
|
||||||
* `__BSS_START__` Must be aligned on a 16-byte boundary.
|
* `__BSS_START__`
|
||||||
* `__BSS_SIZE__`
|
* `__BSS_SIZE__`
|
||||||
* `__COHERENT_RAM_START__` Must be aligned on a page-size boundary.
|
* `__COHERENT_RAM_START__` Must be aligned on a page-size boundary.
|
||||||
* `__COHERENT_RAM_END__` Must be aligned on a page-size boundary.
|
* `__COHERENT_RAM_END__` Must be aligned on a page-size boundary.
|
||||||
|
|
|
@ -43,6 +43,7 @@
|
||||||
#include <stddef.h>
|
#include <stddef.h>
|
||||||
#include <stdint.h>
|
#include <stdint.h>
|
||||||
#include <string.h>
|
#include <string.h>
|
||||||
|
#include <utils.h>
|
||||||
|
|
||||||
/* mbed TLS headers */
|
/* mbed TLS headers */
|
||||||
#include <mbedtls/asn1.h>
|
#include <mbedtls/asn1.h>
|
||||||
|
@ -71,7 +72,7 @@ static void clear_temp_vars(void)
|
||||||
{
|
{
|
||||||
#define ZERO_AND_CLEAN(x) \
|
#define ZERO_AND_CLEAN(x) \
|
||||||
do { \
|
do { \
|
||||||
memset(&x, 0, sizeof(x)); \
|
zeromem(&x, sizeof(x)); \
|
||||||
clean_dcache_range((uintptr_t)&x, sizeof(x)); \
|
clean_dcache_range((uintptr_t)&x, sizeof(x)); \
|
||||||
} while (0);
|
} while (0);
|
||||||
|
|
||||||
|
@ -111,7 +112,7 @@ static int get_ext(const char *oid, void **ext, unsigned int *ext_len)
|
||||||
MBEDTLS_ASN1_SEQUENCE);
|
MBEDTLS_ASN1_SEQUENCE);
|
||||||
|
|
||||||
while (p < end) {
|
while (p < end) {
|
||||||
memset(&extn_oid, 0x0, sizeof(extn_oid));
|
zeromem(&extn_oid, sizeof(extn_oid));
|
||||||
is_critical = 0; /* DEFAULT FALSE */
|
is_critical = 0; /* DEFAULT FALSE */
|
||||||
|
|
||||||
mbedtls_asn1_get_tag(&p, end, &len, MBEDTLS_ASN1_CONSTRUCTED |
|
mbedtls_asn1_get_tag(&p, end, &len, MBEDTLS_ASN1_CONSTRUCTED |
|
||||||
|
|
|
@ -1,5 +1,5 @@
|
||||||
/*
|
/*
|
||||||
* Copyright (c) 2016, ARM Limited and Contributors. All rights reserved.
|
* Copyright (c) 2016-2017, ARM Limited and Contributors. All rights reserved.
|
||||||
*
|
*
|
||||||
* Redistribution and use in source and binary forms, with or without
|
* Redistribution and use in source and binary forms, with or without
|
||||||
* modification, are permitted provided that the following conditions are met:
|
* modification, are permitted provided that the following conditions are met:
|
||||||
|
@ -36,6 +36,7 @@
|
||||||
#include <emmc.h>
|
#include <emmc.h>
|
||||||
#include <errno.h>
|
#include <errno.h>
|
||||||
#include <string.h>
|
#include <string.h>
|
||||||
|
#include <utils.h>
|
||||||
|
|
||||||
static const emmc_ops_t *ops;
|
static const emmc_ops_t *ops;
|
||||||
static unsigned int emmc_ocr_value;
|
static unsigned int emmc_ocr_value;
|
||||||
|
@ -53,7 +54,7 @@ static int emmc_device_state(void)
|
||||||
int ret;
|
int ret;
|
||||||
|
|
||||||
do {
|
do {
|
||||||
memset(&cmd, 0, sizeof(emmc_cmd_t));
|
zeromem(&cmd, sizeof(emmc_cmd_t));
|
||||||
cmd.cmd_idx = EMMC_CMD13;
|
cmd.cmd_idx = EMMC_CMD13;
|
||||||
cmd.cmd_arg = EMMC_FIX_RCA << RCA_SHIFT_OFFSET;
|
cmd.cmd_arg = EMMC_FIX_RCA << RCA_SHIFT_OFFSET;
|
||||||
cmd.resp_type = EMMC_RESPONSE_R1;
|
cmd.resp_type = EMMC_RESPONSE_R1;
|
||||||
|
@ -71,7 +72,7 @@ static void emmc_set_ext_csd(unsigned int ext_cmd, unsigned int value)
|
||||||
emmc_cmd_t cmd;
|
emmc_cmd_t cmd;
|
||||||
int ret, state;
|
int ret, state;
|
||||||
|
|
||||||
memset(&cmd, 0, sizeof(emmc_cmd_t));
|
zeromem(&cmd, sizeof(emmc_cmd_t));
|
||||||
cmd.cmd_idx = EMMC_CMD6;
|
cmd.cmd_idx = EMMC_CMD6;
|
||||||
cmd.cmd_arg = EXTCSD_WRITE_BYTES | EXTCSD_CMD(ext_cmd) |
|
cmd.cmd_arg = EXTCSD_WRITE_BYTES | EXTCSD_CMD(ext_cmd) |
|
||||||
EXTCSD_VALUE(value) | 1;
|
EXTCSD_VALUE(value) | 1;
|
||||||
|
@ -107,14 +108,14 @@ static int emmc_enumerate(int clk, int bus_width)
|
||||||
ops->init();
|
ops->init();
|
||||||
|
|
||||||
/* CMD0: reset to IDLE */
|
/* CMD0: reset to IDLE */
|
||||||
memset(&cmd, 0, sizeof(emmc_cmd_t));
|
zeromem(&cmd, sizeof(emmc_cmd_t));
|
||||||
cmd.cmd_idx = EMMC_CMD0;
|
cmd.cmd_idx = EMMC_CMD0;
|
||||||
ret = ops->send_cmd(&cmd);
|
ret = ops->send_cmd(&cmd);
|
||||||
assert(ret == 0);
|
assert(ret == 0);
|
||||||
|
|
||||||
while (1) {
|
while (1) {
|
||||||
/* CMD1: get OCR register */
|
/* CMD1: get OCR register */
|
||||||
memset(&cmd, 0, sizeof(emmc_cmd_t));
|
zeromem(&cmd, sizeof(emmc_cmd_t));
|
||||||
cmd.cmd_idx = EMMC_CMD1;
|
cmd.cmd_idx = EMMC_CMD1;
|
||||||
cmd.cmd_arg = OCR_SECTOR_MODE | OCR_VDD_MIN_2V7 |
|
cmd.cmd_arg = OCR_SECTOR_MODE | OCR_VDD_MIN_2V7 |
|
||||||
OCR_VDD_MIN_1V7;
|
OCR_VDD_MIN_1V7;
|
||||||
|
@ -127,14 +128,14 @@ static int emmc_enumerate(int clk, int bus_width)
|
||||||
}
|
}
|
||||||
|
|
||||||
/* CMD2: Card Identification */
|
/* CMD2: Card Identification */
|
||||||
memset(&cmd, 0, sizeof(emmc_cmd_t));
|
zeromem(&cmd, sizeof(emmc_cmd_t));
|
||||||
cmd.cmd_idx = EMMC_CMD2;
|
cmd.cmd_idx = EMMC_CMD2;
|
||||||
cmd.resp_type = EMMC_RESPONSE_R2;
|
cmd.resp_type = EMMC_RESPONSE_R2;
|
||||||
ret = ops->send_cmd(&cmd);
|
ret = ops->send_cmd(&cmd);
|
||||||
assert(ret == 0);
|
assert(ret == 0);
|
||||||
|
|
||||||
/* CMD3: Set Relative Address */
|
/* CMD3: Set Relative Address */
|
||||||
memset(&cmd, 0, sizeof(emmc_cmd_t));
|
zeromem(&cmd, sizeof(emmc_cmd_t));
|
||||||
cmd.cmd_idx = EMMC_CMD3;
|
cmd.cmd_idx = EMMC_CMD3;
|
||||||
cmd.cmd_arg = EMMC_FIX_RCA << RCA_SHIFT_OFFSET;
|
cmd.cmd_arg = EMMC_FIX_RCA << RCA_SHIFT_OFFSET;
|
||||||
cmd.resp_type = EMMC_RESPONSE_R1;
|
cmd.resp_type = EMMC_RESPONSE_R1;
|
||||||
|
@ -142,7 +143,7 @@ static int emmc_enumerate(int clk, int bus_width)
|
||||||
assert(ret == 0);
|
assert(ret == 0);
|
||||||
|
|
||||||
/* CMD9: CSD Register */
|
/* CMD9: CSD Register */
|
||||||
memset(&cmd, 0, sizeof(emmc_cmd_t));
|
zeromem(&cmd, sizeof(emmc_cmd_t));
|
||||||
cmd.cmd_idx = EMMC_CMD9;
|
cmd.cmd_idx = EMMC_CMD9;
|
||||||
cmd.cmd_arg = EMMC_FIX_RCA << RCA_SHIFT_OFFSET;
|
cmd.cmd_arg = EMMC_FIX_RCA << RCA_SHIFT_OFFSET;
|
||||||
cmd.resp_type = EMMC_RESPONSE_R2;
|
cmd.resp_type = EMMC_RESPONSE_R2;
|
||||||
|
@ -151,7 +152,7 @@ static int emmc_enumerate(int clk, int bus_width)
|
||||||
memcpy(&emmc_csd, &cmd.resp_data, sizeof(cmd.resp_data));
|
memcpy(&emmc_csd, &cmd.resp_data, sizeof(cmd.resp_data));
|
||||||
|
|
||||||
/* CMD7: Select Card */
|
/* CMD7: Select Card */
|
||||||
memset(&cmd, 0, sizeof(emmc_cmd_t));
|
zeromem(&cmd, sizeof(emmc_cmd_t));
|
||||||
cmd.cmd_idx = EMMC_CMD7;
|
cmd.cmd_idx = EMMC_CMD7;
|
||||||
cmd.cmd_arg = EMMC_FIX_RCA << RCA_SHIFT_OFFSET;
|
cmd.cmd_arg = EMMC_FIX_RCA << RCA_SHIFT_OFFSET;
|
||||||
cmd.resp_type = EMMC_RESPONSE_R1;
|
cmd.resp_type = EMMC_RESPONSE_R1;
|
||||||
|
@ -181,7 +182,7 @@ size_t emmc_read_blocks(int lba, uintptr_t buf, size_t size)
|
||||||
assert(ret == 0);
|
assert(ret == 0);
|
||||||
|
|
||||||
if (is_cmd23_enabled()) {
|
if (is_cmd23_enabled()) {
|
||||||
memset(&cmd, 0, sizeof(emmc_cmd_t));
|
zeromem(&cmd, sizeof(emmc_cmd_t));
|
||||||
/* set block count */
|
/* set block count */
|
||||||
cmd.cmd_idx = EMMC_CMD23;
|
cmd.cmd_idx = EMMC_CMD23;
|
||||||
cmd.cmd_arg = size / EMMC_BLOCK_SIZE;
|
cmd.cmd_arg = size / EMMC_BLOCK_SIZE;
|
||||||
|
@ -189,7 +190,7 @@ size_t emmc_read_blocks(int lba, uintptr_t buf, size_t size)
|
||||||
ret = ops->send_cmd(&cmd);
|
ret = ops->send_cmd(&cmd);
|
||||||
assert(ret == 0);
|
assert(ret == 0);
|
||||||
|
|
||||||
memset(&cmd, 0, sizeof(emmc_cmd_t));
|
zeromem(&cmd, sizeof(emmc_cmd_t));
|
||||||
cmd.cmd_idx = EMMC_CMD18;
|
cmd.cmd_idx = EMMC_CMD18;
|
||||||
} else {
|
} else {
|
||||||
if (size > EMMC_BLOCK_SIZE)
|
if (size > EMMC_BLOCK_SIZE)
|
||||||
|
@ -213,7 +214,7 @@ size_t emmc_read_blocks(int lba, uintptr_t buf, size_t size)
|
||||||
|
|
||||||
if (is_cmd23_enabled() == 0) {
|
if (is_cmd23_enabled() == 0) {
|
||||||
if (size > EMMC_BLOCK_SIZE) {
|
if (size > EMMC_BLOCK_SIZE) {
|
||||||
memset(&cmd, 0, sizeof(emmc_cmd_t));
|
zeromem(&cmd, sizeof(emmc_cmd_t));
|
||||||
cmd.cmd_idx = EMMC_CMD12;
|
cmd.cmd_idx = EMMC_CMD12;
|
||||||
ret = ops->send_cmd(&cmd);
|
ret = ops->send_cmd(&cmd);
|
||||||
assert(ret == 0);
|
assert(ret == 0);
|
||||||
|
@ -240,17 +241,17 @@ size_t emmc_write_blocks(int lba, const uintptr_t buf, size_t size)
|
||||||
|
|
||||||
if (is_cmd23_enabled()) {
|
if (is_cmd23_enabled()) {
|
||||||
/* set block count */
|
/* set block count */
|
||||||
memset(&cmd, 0, sizeof(emmc_cmd_t));
|
zeromem(&cmd, sizeof(emmc_cmd_t));
|
||||||
cmd.cmd_idx = EMMC_CMD23;
|
cmd.cmd_idx = EMMC_CMD23;
|
||||||
cmd.cmd_arg = size / EMMC_BLOCK_SIZE;
|
cmd.cmd_arg = size / EMMC_BLOCK_SIZE;
|
||||||
cmd.resp_type = EMMC_RESPONSE_R1;
|
cmd.resp_type = EMMC_RESPONSE_R1;
|
||||||
ret = ops->send_cmd(&cmd);
|
ret = ops->send_cmd(&cmd);
|
||||||
assert(ret == 0);
|
assert(ret == 0);
|
||||||
|
|
||||||
memset(&cmd, 0, sizeof(emmc_cmd_t));
|
zeromem(&cmd, sizeof(emmc_cmd_t));
|
||||||
cmd.cmd_idx = EMMC_CMD25;
|
cmd.cmd_idx = EMMC_CMD25;
|
||||||
} else {
|
} else {
|
||||||
memset(&cmd, 0, sizeof(emmc_cmd_t));
|
zeromem(&cmd, sizeof(emmc_cmd_t));
|
||||||
if (size > EMMC_BLOCK_SIZE)
|
if (size > EMMC_BLOCK_SIZE)
|
||||||
cmd.cmd_idx = EMMC_CMD25;
|
cmd.cmd_idx = EMMC_CMD25;
|
||||||
else
|
else
|
||||||
|
@ -272,7 +273,7 @@ size_t emmc_write_blocks(int lba, const uintptr_t buf, size_t size)
|
||||||
|
|
||||||
if (is_cmd23_enabled() == 0) {
|
if (is_cmd23_enabled() == 0) {
|
||||||
if (size > EMMC_BLOCK_SIZE) {
|
if (size > EMMC_BLOCK_SIZE) {
|
||||||
memset(&cmd, 0, sizeof(emmc_cmd_t));
|
zeromem(&cmd, sizeof(emmc_cmd_t));
|
||||||
cmd.cmd_idx = EMMC_CMD12;
|
cmd.cmd_idx = EMMC_CMD12;
|
||||||
ret = ops->send_cmd(&cmd);
|
ret = ops->send_cmd(&cmd);
|
||||||
assert(ret == 0);
|
assert(ret == 0);
|
||||||
|
@ -291,21 +292,21 @@ size_t emmc_erase_blocks(int lba, size_t size)
|
||||||
assert(ops != 0);
|
assert(ops != 0);
|
||||||
assert((size != 0) && ((size % EMMC_BLOCK_SIZE) == 0));
|
assert((size != 0) && ((size % EMMC_BLOCK_SIZE) == 0));
|
||||||
|
|
||||||
memset(&cmd, 0, sizeof(emmc_cmd_t));
|
zeromem(&cmd, sizeof(emmc_cmd_t));
|
||||||
cmd.cmd_idx = EMMC_CMD35;
|
cmd.cmd_idx = EMMC_CMD35;
|
||||||
cmd.cmd_arg = lba;
|
cmd.cmd_arg = lba;
|
||||||
cmd.resp_type = EMMC_RESPONSE_R1;
|
cmd.resp_type = EMMC_RESPONSE_R1;
|
||||||
ret = ops->send_cmd(&cmd);
|
ret = ops->send_cmd(&cmd);
|
||||||
assert(ret == 0);
|
assert(ret == 0);
|
||||||
|
|
||||||
memset(&cmd, 0, sizeof(emmc_cmd_t));
|
zeromem(&cmd, sizeof(emmc_cmd_t));
|
||||||
cmd.cmd_idx = EMMC_CMD36;
|
cmd.cmd_idx = EMMC_CMD36;
|
||||||
cmd.cmd_arg = lba + (size / EMMC_BLOCK_SIZE) - 1;
|
cmd.cmd_arg = lba + (size / EMMC_BLOCK_SIZE) - 1;
|
||||||
cmd.resp_type = EMMC_RESPONSE_R1;
|
cmd.resp_type = EMMC_RESPONSE_R1;
|
||||||
ret = ops->send_cmd(&cmd);
|
ret = ops->send_cmd(&cmd);
|
||||||
assert(ret == 0);
|
assert(ret == 0);
|
||||||
|
|
||||||
memset(&cmd, 0, sizeof(emmc_cmd_t));
|
zeromem(&cmd, sizeof(emmc_cmd_t));
|
||||||
cmd.cmd_idx = EMMC_CMD38;
|
cmd.cmd_idx = EMMC_CMD38;
|
||||||
cmd.resp_type = EMMC_RESPONSE_R1B;
|
cmd.resp_type = EMMC_RESPONSE_R1B;
|
||||||
ret = ops->send_cmd(&cmd);
|
ret = ops->send_cmd(&cmd);
|
||||||
|
|
|
@ -1,5 +1,5 @@
|
||||||
/*
|
/*
|
||||||
* Copyright (c) 2016, ARM Limited and Contributors. All rights reserved.
|
* Copyright (c) 2016-2017, ARM Limited and Contributors. All rights reserved.
|
||||||
*
|
*
|
||||||
* Redistribution and use in source and binary forms, with or without
|
* Redistribution and use in source and binary forms, with or without
|
||||||
* modification, are permitted provided that the following conditions are met:
|
* modification, are permitted provided that the following conditions are met:
|
||||||
|
@ -36,6 +36,7 @@
|
||||||
#include <io_storage.h>
|
#include <io_storage.h>
|
||||||
#include <platform_def.h>
|
#include <platform_def.h>
|
||||||
#include <string.h>
|
#include <string.h>
|
||||||
|
#include <utils.h>
|
||||||
|
|
||||||
typedef struct {
|
typedef struct {
|
||||||
io_block_dev_spec_t *dev_spec;
|
io_block_dev_spec_t *dev_spec;
|
||||||
|
@ -135,8 +136,8 @@ static int free_dev_info(io_dev_info_t *dev_info)
|
||||||
result = find_first_block_state(state->dev_spec, &index);
|
result = find_first_block_state(state->dev_spec, &index);
|
||||||
if (result == 0) {
|
if (result == 0) {
|
||||||
/* free if device info is valid */
|
/* free if device info is valid */
|
||||||
memset(state, 0, sizeof(block_dev_state_t));
|
zeromem(state, sizeof(block_dev_state_t));
|
||||||
memset(dev_info, 0, sizeof(io_dev_info_t));
|
zeromem(dev_info, sizeof(io_dev_info_t));
|
||||||
--block_dev_count;
|
--block_dev_count;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -1,5 +1,5 @@
|
||||||
/*
|
/*
|
||||||
* Copyright (c) 2014, ARM Limited and Contributors. All rights reserved.
|
* Copyright (c) 2014-2017, ARM Limited and Contributors. All rights reserved.
|
||||||
*
|
*
|
||||||
* Redistribution and use in source and binary forms, with or without
|
* Redistribution and use in source and binary forms, with or without
|
||||||
* modification, are permitted provided that the following conditions are met:
|
* modification, are permitted provided that the following conditions are met:
|
||||||
|
@ -40,6 +40,7 @@
|
||||||
#include <platform_def.h>
|
#include <platform_def.h>
|
||||||
#include <stdint.h>
|
#include <stdint.h>
|
||||||
#include <string.h>
|
#include <string.h>
|
||||||
|
#include <utils.h>
|
||||||
#include <uuid.h>
|
#include <uuid.h>
|
||||||
|
|
||||||
/* Useful for printing UUIDs when debugging.*/
|
/* Useful for printing UUIDs when debugging.*/
|
||||||
|
@ -351,7 +352,7 @@ static int fip_file_close(io_entity_t *entity)
|
||||||
* If we had malloc() we would free() here.
|
* If we had malloc() we would free() here.
|
||||||
*/
|
*/
|
||||||
if (current_file.entry.offset_address != 0) {
|
if (current_file.entry.offset_address != 0) {
|
||||||
memset(¤t_file, 0, sizeof(current_file));
|
zeromem(¤t_file, sizeof(current_file));
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Clear the Entity info. */
|
/* Clear the Entity info. */
|
||||||
|
|
|
@ -1,5 +1,5 @@
|
||||||
/*
|
/*
|
||||||
* Copyright (c) 2014, ARM Limited and Contributors. All rights reserved.
|
* Copyright (c) 2014-2017, ARM Limited and Contributors. All rights reserved.
|
||||||
*
|
*
|
||||||
* Redistribution and use in source and binary forms, with or without
|
* Redistribution and use in source and binary forms, with or without
|
||||||
* modification, are permitted provided that the following conditions are met:
|
* modification, are permitted provided that the following conditions are met:
|
||||||
|
@ -33,6 +33,7 @@
|
||||||
#include <io_driver.h>
|
#include <io_driver.h>
|
||||||
#include <io_storage.h>
|
#include <io_storage.h>
|
||||||
#include <string.h>
|
#include <string.h>
|
||||||
|
#include <utils.h>
|
||||||
|
|
||||||
/* As we need to be able to keep state for seek, only one file can be open
|
/* As we need to be able to keep state for seek, only one file can be open
|
||||||
* at a time. Make this a structure and point to the entity->info. When we
|
* at a time. Make this a structure and point to the entity->info. When we
|
||||||
|
@ -231,7 +232,7 @@ static int memmap_block_close(io_entity_t *entity)
|
||||||
entity->info = 0;
|
entity->info = 0;
|
||||||
|
|
||||||
/* This would be a mem free() if we had malloc.*/
|
/* This would be a mem free() if we had malloc.*/
|
||||||
memset((void *)¤t_file, 0, sizeof(current_file));
|
zeromem((void *)¤t_file, sizeof(current_file));
|
||||||
|
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
|
@ -1,5 +1,5 @@
|
||||||
/*
|
/*
|
||||||
* Copyright (c) 2016, ARM Limited and Contributors. All rights reserved.
|
* Copyright (c) 2016-2017, ARM Limited and Contributors. All rights reserved.
|
||||||
*
|
*
|
||||||
* Redistribution and use in source and binary forms, with or without
|
* Redistribution and use in source and binary forms, with or without
|
||||||
* modification, are permitted provided that the following conditions are met:
|
* modification, are permitted provided that the following conditions are met:
|
||||||
|
@ -33,6 +33,7 @@
|
||||||
#include <errno.h>
|
#include <errno.h>
|
||||||
#include <gpt.h>
|
#include <gpt.h>
|
||||||
#include <string.h>
|
#include <string.h>
|
||||||
|
#include <utils.h>
|
||||||
|
|
||||||
static int unicode_to_ascii(unsigned short *str_in, unsigned char *str_out)
|
static int unicode_to_ascii(unsigned short *str_in, unsigned char *str_out)
|
||||||
{
|
{
|
||||||
|
@ -65,7 +66,7 @@ int parse_gpt_entry(gpt_entry_t *gpt_entry, partition_entry_t *entry)
|
||||||
return -EINVAL;
|
return -EINVAL;
|
||||||
}
|
}
|
||||||
|
|
||||||
memset(entry, 0, sizeof(partition_entry_t));
|
zeromem(entry, sizeof(partition_entry_t));
|
||||||
result = unicode_to_ascii(gpt_entry->name, (uint8_t *)entry->name);
|
result = unicode_to_ascii(gpt_entry->name, (uint8_t *)entry->name);
|
||||||
if (result != 0) {
|
if (result != 0) {
|
||||||
return result;
|
return result;
|
||||||
|
|
|
@ -1,5 +1,5 @@
|
||||||
/*
|
/*
|
||||||
* Copyright (c) 2015-2016, ARM Limited and Contributors. All rights reserved.
|
* Copyright (c) 2015-2017, ARM Limited and Contributors. All rights reserved.
|
||||||
*
|
*
|
||||||
* Redistribution and use in source and binary forms, with or without
|
* Redistribution and use in source and binary forms, with or without
|
||||||
* modification, are permitted provided that the following conditions are met:
|
* modification, are permitted provided that the following conditions are met:
|
||||||
|
@ -252,12 +252,12 @@
|
||||||
|
|
||||||
ldr x0, =__BSS_START__
|
ldr x0, =__BSS_START__
|
||||||
ldr x1, =__BSS_SIZE__
|
ldr x1, =__BSS_SIZE__
|
||||||
bl zeromem16
|
bl zeromem
|
||||||
|
|
||||||
#if USE_COHERENT_MEM
|
#if USE_COHERENT_MEM
|
||||||
ldr x0, =__COHERENT_RAM_START__
|
ldr x0, =__COHERENT_RAM_START__
|
||||||
ldr x1, =__COHERENT_RAM_UNALIGNED_SIZE__
|
ldr x1, =__COHERENT_RAM_UNALIGNED_SIZE__
|
||||||
bl zeromem16
|
bl zeromem
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
#ifdef IMAGE_BL1
|
#ifdef IMAGE_BL1
|
||||||
|
|
|
@ -80,4 +80,35 @@
|
||||||
# define ULL(_x) (_x##ull)
|
# define ULL(_x) (_x##ull)
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
|
/*
|
||||||
|
* C code should be put in this part of the header to avoid breaking ASM files
|
||||||
|
* or linker scripts including it.
|
||||||
|
*/
|
||||||
|
#if !(defined(__LINKER__) || defined(__ASSEMBLY__))
|
||||||
|
|
||||||
|
#include <types.h>
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Fill a region of normal memory of size "length" in bytes with zero bytes.
|
||||||
|
*
|
||||||
|
* WARNING: This function can only operate on normal memory. This means that
|
||||||
|
* the MMU must be enabled when using this function. Otherwise, use
|
||||||
|
* zeromem.
|
||||||
|
*/
|
||||||
|
void zero_normalmem(void *mem, u_register_t length);
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Fill a region of memory of size "length" in bytes with null bytes.
|
||||||
|
*
|
||||||
|
* Unlike zero_normalmem, this function has no restriction on the type of
|
||||||
|
* memory targeted and can be used for any device memory as well as normal
|
||||||
|
* memory. This function must be used instead of zero_normalmem when MMU is
|
||||||
|
* disabled.
|
||||||
|
*
|
||||||
|
* NOTE: When data cache and MMU are enabled, prefer zero_normalmem for faster
|
||||||
|
* zeroing.
|
||||||
|
*/
|
||||||
|
void zeromem(void *mem, u_register_t length);
|
||||||
|
#endif /* !(defined(__LINKER__) || defined(__ASSEMBLY__)) */
|
||||||
|
|
||||||
#endif /* __UTILS_H__ */
|
#endif /* __UTILS_H__ */
|
||||||
|
|
|
@ -1,5 +1,5 @@
|
||||||
/*
|
/*
|
||||||
* Copyright (c) 2016, ARM Limited and Contributors. All rights reserved.
|
* Copyright (c) 2016-2017, ARM Limited and Contributors. All rights reserved.
|
||||||
*
|
*
|
||||||
* Redistribution and use in source and binary forms, with or without
|
* Redistribution and use in source and binary forms, with or without
|
||||||
* modification, are permitted provided that the following conditions are met:
|
* modification, are permitted provided that the following conditions are met:
|
||||||
|
@ -34,6 +34,7 @@
|
||||||
|
|
||||||
.globl smc
|
.globl smc
|
||||||
.globl zeromem
|
.globl zeromem
|
||||||
|
.globl zero_normalmem
|
||||||
.globl memcpy4
|
.globl memcpy4
|
||||||
.globl disable_mmu_icache_secure
|
.globl disable_mmu_icache_secure
|
||||||
.globl disable_mmu_secure
|
.globl disable_mmu_secure
|
||||||
|
@ -50,30 +51,108 @@ func smc
|
||||||
endfunc smc
|
endfunc smc
|
||||||
|
|
||||||
/* -----------------------------------------------------------------------
|
/* -----------------------------------------------------------------------
|
||||||
* void zeromem(void *mem, unsigned int length);
|
* void zeromem(void *mem, unsigned int length)
|
||||||
|
*
|
||||||
|
* Initialise a region in normal memory to 0. This functions complies with the
|
||||||
|
* AAPCS and can be called from C code.
|
||||||
*
|
*
|
||||||
* Initialise a memory region to 0.
|
|
||||||
* The memory address and length must be 4-byte aligned.
|
|
||||||
* -----------------------------------------------------------------------
|
* -----------------------------------------------------------------------
|
||||||
*/
|
*/
|
||||||
func zeromem
|
func zeromem
|
||||||
#if ASM_ASSERTION
|
/*
|
||||||
tst r0, #0x3
|
* Readable names for registers
|
||||||
ASM_ASSERT(eq)
|
*
|
||||||
tst r1, #0x3
|
* Registers r0, r1 and r2 are also set by zeromem which
|
||||||
ASM_ASSERT(eq)
|
* branches into the fallback path directly, so cursor, length and
|
||||||
#endif
|
* stop_address should not be retargeted to other registers.
|
||||||
add r2, r0, r1
|
*/
|
||||||
mov r1, #0
|
cursor .req r0 /* Start address and then current address */
|
||||||
z_loop:
|
length .req r1 /* Length in bytes of the region to zero out */
|
||||||
cmp r2, r0
|
/*
|
||||||
beq z_end
|
* Reusing the r1 register as length is only used at the beginning of
|
||||||
str r1, [r0], #4
|
* the function.
|
||||||
b z_loop
|
*/
|
||||||
z_end:
|
stop_address .req r1 /* Address past the last zeroed byte */
|
||||||
|
zeroreg1 .req r2 /* Source register filled with 0 */
|
||||||
|
zeroreg2 .req r3 /* Source register filled with 0 */
|
||||||
|
tmp .req r12 /* Temporary scratch register */
|
||||||
|
|
||||||
|
mov zeroreg1, #0
|
||||||
|
|
||||||
|
/* stop_address is the address past the last to zero */
|
||||||
|
add stop_address, cursor, length
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Length cannot be used anymore as it shares the same register with
|
||||||
|
* stop_address.
|
||||||
|
*/
|
||||||
|
.unreq length
|
||||||
|
|
||||||
|
/*
|
||||||
|
* If the start address is already aligned to 8 bytes, skip this loop.
|
||||||
|
*/
|
||||||
|
tst cursor, #(8-1)
|
||||||
|
beq .Lzeromem_8bytes_aligned
|
||||||
|
|
||||||
|
/* Calculate the next address aligned to 8 bytes */
|
||||||
|
orr tmp, cursor, #(8-1)
|
||||||
|
adds tmp, tmp, #1
|
||||||
|
/* If it overflows, fallback to byte per byte zeroing */
|
||||||
|
beq .Lzeromem_1byte_aligned
|
||||||
|
/* If the next aligned address is after the stop address, fall back */
|
||||||
|
cmp tmp, stop_address
|
||||||
|
bhs .Lzeromem_1byte_aligned
|
||||||
|
|
||||||
|
/* zero byte per byte */
|
||||||
|
1:
|
||||||
|
strb zeroreg1, [cursor], #1
|
||||||
|
cmp cursor, tmp
|
||||||
|
bne 1b
|
||||||
|
|
||||||
|
/* zero 8 bytes at a time */
|
||||||
|
.Lzeromem_8bytes_aligned:
|
||||||
|
|
||||||
|
/* Calculate the last 8 bytes aligned address. */
|
||||||
|
bic tmp, stop_address, #(8-1)
|
||||||
|
|
||||||
|
cmp cursor, tmp
|
||||||
|
bhs 2f
|
||||||
|
|
||||||
|
mov zeroreg2, #0
|
||||||
|
1:
|
||||||
|
stmia cursor!, {zeroreg1, zeroreg2}
|
||||||
|
cmp cursor, tmp
|
||||||
|
blo 1b
|
||||||
|
2:
|
||||||
|
|
||||||
|
/* zero byte per byte */
|
||||||
|
.Lzeromem_1byte_aligned:
|
||||||
|
cmp cursor, stop_address
|
||||||
|
beq 2f
|
||||||
|
1:
|
||||||
|
strb zeroreg1, [cursor], #1
|
||||||
|
cmp cursor, stop_address
|
||||||
|
bne 1b
|
||||||
|
2:
|
||||||
bx lr
|
bx lr
|
||||||
|
|
||||||
|
.unreq cursor
|
||||||
|
/*
|
||||||
|
* length is already unreq'ed to reuse the register for another
|
||||||
|
* variable.
|
||||||
|
*/
|
||||||
|
.unreq stop_address
|
||||||
|
.unreq zeroreg1
|
||||||
|
.unreq zeroreg2
|
||||||
|
.unreq tmp
|
||||||
endfunc zeromem
|
endfunc zeromem
|
||||||
|
|
||||||
|
/*
|
||||||
|
* AArch32 does not have special ways of zeroing normal memory as AArch64 does
|
||||||
|
* using the DC ZVA instruction, so we just alias zero_normalmem to zeromem.
|
||||||
|
*/
|
||||||
|
.equ zero_normalmem, zeromem
|
||||||
|
|
||||||
/* --------------------------------------------------------------------------
|
/* --------------------------------------------------------------------------
|
||||||
* void memcpy4(void *dest, const void *src, unsigned int length)
|
* void memcpy4(void *dest, const void *src, unsigned int length)
|
||||||
*
|
*
|
||||||
|
|
|
@ -1,5 +1,5 @@
|
||||||
/*
|
/*
|
||||||
* Copyright (c) 2013-2014, ARM Limited and Contributors. All rights reserved.
|
* Copyright (c) 2013-2017, ARM Limited and Contributors. All rights reserved.
|
||||||
*
|
*
|
||||||
* Redistribution and use in source and binary forms, with or without
|
* Redistribution and use in source and binary forms, with or without
|
||||||
* modification, are permitted provided that the following conditions are met:
|
* modification, are permitted provided that the following conditions are met:
|
||||||
|
@ -37,6 +37,8 @@
|
||||||
.globl eret
|
.globl eret
|
||||||
.globl smc
|
.globl smc
|
||||||
|
|
||||||
|
.globl zero_normalmem
|
||||||
|
.globl zeromem
|
||||||
.globl zeromem16
|
.globl zeromem16
|
||||||
.globl memcpy16
|
.globl memcpy16
|
||||||
|
|
||||||
|
@ -80,31 +82,358 @@ endfunc smc
|
||||||
*
|
*
|
||||||
* Initialise a memory region to 0.
|
* Initialise a memory region to 0.
|
||||||
* The memory address must be 16-byte aligned.
|
* The memory address must be 16-byte aligned.
|
||||||
|
* NOTE: This function is deprecated and zeromem should be used instead.
|
||||||
* -----------------------------------------------------------------------
|
* -----------------------------------------------------------------------
|
||||||
*/
|
*/
|
||||||
func zeromem16
|
.equ zeromem16, zeromem
|
||||||
#if ASM_ASSERTION
|
|
||||||
tst x0, #0xf
|
|
||||||
ASM_ASSERT(eq)
|
|
||||||
#endif
|
|
||||||
add x2, x0, x1
|
|
||||||
/* zero 16 bytes at a time */
|
|
||||||
z_loop16:
|
|
||||||
sub x3, x2, x0
|
|
||||||
cmp x3, #16
|
|
||||||
b.lt z_loop1
|
|
||||||
stp xzr, xzr, [x0], #16
|
|
||||||
b z_loop16
|
|
||||||
/* zero byte per byte */
|
|
||||||
z_loop1:
|
|
||||||
cmp x0, x2
|
|
||||||
b.eq z_end
|
|
||||||
strb wzr, [x0], #1
|
|
||||||
b z_loop1
|
|
||||||
z_end:
|
|
||||||
ret
|
|
||||||
endfunc zeromem16
|
|
||||||
|
|
||||||
|
/* -----------------------------------------------------------------------
|
||||||
|
* void zero_normalmem(void *mem, unsigned int length);
|
||||||
|
*
|
||||||
|
* Initialise a region in normal memory to 0. This functions complies with the
|
||||||
|
* AAPCS and can be called from C code.
|
||||||
|
*
|
||||||
|
* NOTE: MMU must be enabled when using this function as it can only operate on
|
||||||
|
* normal memory. It is intended to be mainly used from C code when MMU
|
||||||
|
* is usually enabled.
|
||||||
|
* -----------------------------------------------------------------------
|
||||||
|
*/
|
||||||
|
.equ zero_normalmem, zeromem_dczva
|
||||||
|
|
||||||
|
/* -----------------------------------------------------------------------
|
||||||
|
* void zeromem(void *mem, unsigned int length);
|
||||||
|
*
|
||||||
|
* Initialise a region of device memory to 0. This functions complies with the
|
||||||
|
* AAPCS and can be called from C code.
|
||||||
|
*
|
||||||
|
* NOTE: When data caches and MMU are enabled, zero_normalmem can usually be
|
||||||
|
* used instead for faster zeroing.
|
||||||
|
*
|
||||||
|
* -----------------------------------------------------------------------
|
||||||
|
*/
|
||||||
|
func zeromem
|
||||||
|
/* x2 is the address past the last zeroed address */
|
||||||
|
add x2, x0, x1
|
||||||
|
/*
|
||||||
|
* Uses the fallback path that does not use DC ZVA instruction and
|
||||||
|
* therefore does not need enabled MMU
|
||||||
|
*/
|
||||||
|
b .Lzeromem_dczva_fallback_entry
|
||||||
|
endfunc zeromem
|
||||||
|
|
||||||
|
/* -----------------------------------------------------------------------
|
||||||
|
* void zeromem_dczva(void *mem, unsigned int length);
|
||||||
|
*
|
||||||
|
* Fill a region of normal memory of size "length" in bytes with null bytes.
|
||||||
|
* MMU must be enabled and the memory be of
|
||||||
|
* normal type. This is because this function internally uses the DC ZVA
|
||||||
|
* instruction, which generates an Alignment fault if used on any type of
|
||||||
|
* Device memory (see section D3.4.9 of the ARMv8 ARM, issue k). When the MMU
|
||||||
|
* is disabled, all memory behaves like Device-nGnRnE memory (see section
|
||||||
|
* D4.2.8), hence the requirement on the MMU being enabled.
|
||||||
|
* NOTE: The code assumes that the block size as defined in DCZID_EL0
|
||||||
|
* register is at least 16 bytes.
|
||||||
|
*
|
||||||
|
* -----------------------------------------------------------------------
|
||||||
|
*/
|
||||||
|
func zeromem_dczva
|
||||||
|
|
||||||
|
/*
|
||||||
|
* The function consists of a series of loops that zero memory one byte
|
||||||
|
* at a time, 16 bytes at a time or using the DC ZVA instruction to
|
||||||
|
* zero aligned block of bytes, which is assumed to be more than 16.
|
||||||
|
* In the case where the DC ZVA instruction cannot be used or if the
|
||||||
|
* first 16 bytes loop would overflow, there is fallback path that does
|
||||||
|
* not use DC ZVA.
|
||||||
|
* Note: The fallback path is also used by the zeromem function that
|
||||||
|
* branches to it directly.
|
||||||
|
*
|
||||||
|
* +---------+ zeromem_dczva
|
||||||
|
* | entry |
|
||||||
|
* +----+----+
|
||||||
|
* |
|
||||||
|
* v
|
||||||
|
* +---------+
|
||||||
|
* | checks |>o-------+ (If any check fails, fallback)
|
||||||
|
* +----+----+ |
|
||||||
|
* | |---------------+
|
||||||
|
* v | Fallback path |
|
||||||
|
* +------+------+ |---------------+
|
||||||
|
* | 1 byte loop | |
|
||||||
|
* +------+------+ .Lzeromem_dczva_initial_1byte_aligned_end
|
||||||
|
* | |
|
||||||
|
* v |
|
||||||
|
* +-------+-------+ |
|
||||||
|
* | 16 bytes loop | |
|
||||||
|
* +-------+-------+ |
|
||||||
|
* | |
|
||||||
|
* v |
|
||||||
|
* +------+------+ .Lzeromem_dczva_blocksize_aligned
|
||||||
|
* | DC ZVA loop | |
|
||||||
|
* +------+------+ |
|
||||||
|
* +--------+ | |
|
||||||
|
* | | | |
|
||||||
|
* | v v |
|
||||||
|
* | +-------+-------+ .Lzeromem_dczva_final_16bytes_aligned
|
||||||
|
* | | 16 bytes loop | |
|
||||||
|
* | +-------+-------+ |
|
||||||
|
* | | |
|
||||||
|
* | v |
|
||||||
|
* | +------+------+ .Lzeromem_dczva_final_1byte_aligned
|
||||||
|
* | | 1 byte loop | |
|
||||||
|
* | +-------------+ |
|
||||||
|
* | | |
|
||||||
|
* | v |
|
||||||
|
* | +---+--+ |
|
||||||
|
* | | exit | |
|
||||||
|
* | +------+ |
|
||||||
|
* | |
|
||||||
|
* | +--------------+ +------------------+ zeromem
|
||||||
|
* | | +----------------| zeromem function |
|
||||||
|
* | | | +------------------+
|
||||||
|
* | v v
|
||||||
|
* | +-------------+ .Lzeromem_dczva_fallback_entry
|
||||||
|
* | | 1 byte loop |
|
||||||
|
* | +------+------+
|
||||||
|
* | |
|
||||||
|
* +-----------+
|
||||||
|
*/
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Readable names for registers
|
||||||
|
*
|
||||||
|
* Registers x0, x1 and x2 are also set by zeromem which
|
||||||
|
* branches into the fallback path directly, so cursor, length and
|
||||||
|
* stop_address should not be retargeted to other registers.
|
||||||
|
*/
|
||||||
|
cursor .req x0 /* Start address and then current address */
|
||||||
|
length .req x1 /* Length in bytes of the region to zero out */
|
||||||
|
/* Reusing x1 as length is never used after block_mask is set */
|
||||||
|
block_mask .req x1 /* Bitmask of the block size read in DCZID_EL0 */
|
||||||
|
stop_address .req x2 /* Address past the last zeroed byte */
|
||||||
|
block_size .req x3 /* Size of a block in bytes as read in DCZID_EL0 */
|
||||||
|
tmp1 .req x4
|
||||||
|
tmp2 .req x5
|
||||||
|
|
||||||
|
#if ASM_ASSERTION
|
||||||
|
/*
|
||||||
|
* Check for M bit (MMU enabled) of the current SCTLR_EL(1|3)
|
||||||
|
* register value and panic if the MMU is disabled.
|
||||||
|
*/
|
||||||
|
#if defined(IMAGE_BL1) || defined(IMAGE_BL31)
|
||||||
|
mrs tmp1, sctlr_el3
|
||||||
|
#else
|
||||||
|
mrs tmp1, sctlr_el1
|
||||||
|
#endif
|
||||||
|
|
||||||
|
tst tmp1, #SCTLR_M_BIT
|
||||||
|
ASM_ASSERT(ne)
|
||||||
|
#endif /* ASM_ASSERTION */
|
||||||
|
|
||||||
|
/* stop_address is the address past the last to zero */
|
||||||
|
add stop_address, cursor, length
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Get block_size = (log2(<block size>) >> 2) (see encoding of
|
||||||
|
* dczid_el0 reg)
|
||||||
|
*/
|
||||||
|
mrs block_size, dczid_el0
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Select the 4 lowest bits and convert the extracted log2(<block size
|
||||||
|
* in words>) to <block size in bytes>
|
||||||
|
*/
|
||||||
|
ubfx block_size, block_size, #0, #4
|
||||||
|
mov tmp2, #(1 << 2)
|
||||||
|
lsl block_size, tmp2, block_size
|
||||||
|
|
||||||
|
#if ASM_ASSERTION
|
||||||
|
/*
|
||||||
|
* Assumes block size is at least 16 bytes to avoid manual realignment
|
||||||
|
* of the cursor at the end of the DCZVA loop.
|
||||||
|
*/
|
||||||
|
cmp block_size, #16
|
||||||
|
ASM_ASSERT(hs)
|
||||||
|
#endif
|
||||||
|
/*
|
||||||
|
* Not worth doing all the setup for a region less than a block and
|
||||||
|
* protects against zeroing a whole block when the area to zero is
|
||||||
|
* smaller than that. Also, as it is assumed that the block size is at
|
||||||
|
* least 16 bytes, this also protects the initial aligning loops from
|
||||||
|
* trying to zero 16 bytes when length is less than 16.
|
||||||
|
*/
|
||||||
|
cmp length, block_size
|
||||||
|
b.lo .Lzeromem_dczva_fallback_entry
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Calculate the bitmask of the block alignment. It will never
|
||||||
|
* underflow as the block size is between 4 bytes and 2kB.
|
||||||
|
* block_mask = block_size - 1
|
||||||
|
*/
|
||||||
|
sub block_mask, block_size, #1
|
||||||
|
|
||||||
|
/*
|
||||||
|
* length alias should not be used after this point unless it is
|
||||||
|
* defined as a register other than block_mask's.
|
||||||
|
*/
|
||||||
|
.unreq length
|
||||||
|
|
||||||
|
/*
|
||||||
|
* If the start address is already aligned to zero block size, go
|
||||||
|
* straight to the cache zeroing loop. This is safe because at this
|
||||||
|
* point, the length cannot be smaller than a block size.
|
||||||
|
*/
|
||||||
|
tst cursor, block_mask
|
||||||
|
b.eq .Lzeromem_dczva_blocksize_aligned
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Calculate the first block-size-aligned address. It is assumed that
|
||||||
|
* the zero block size is at least 16 bytes. This address is the last
|
||||||
|
* address of this initial loop.
|
||||||
|
*/
|
||||||
|
orr tmp1, cursor, block_mask
|
||||||
|
add tmp1, tmp1, #1
|
||||||
|
|
||||||
|
/*
|
||||||
|
* If the addition overflows, skip the cache zeroing loops. This is
|
||||||
|
* quite unlikely however.
|
||||||
|
*/
|
||||||
|
cbz tmp1, .Lzeromem_dczva_fallback_entry
|
||||||
|
|
||||||
|
/*
|
||||||
|
* If the first block-size-aligned address is past the last address,
|
||||||
|
* fallback to the simpler code.
|
||||||
|
*/
|
||||||
|
cmp tmp1, stop_address
|
||||||
|
b.hi .Lzeromem_dczva_fallback_entry
|
||||||
|
|
||||||
|
/*
|
||||||
|
* If the start address is already aligned to 16 bytes, skip this loop.
|
||||||
|
* It is safe to do this because tmp1 (the stop address of the initial
|
||||||
|
* 16 bytes loop) will never be greater than the final stop address.
|
||||||
|
*/
|
||||||
|
tst cursor, #0xf
|
||||||
|
b.eq .Lzeromem_dczva_initial_1byte_aligned_end
|
||||||
|
|
||||||
|
/* Calculate the next address aligned to 16 bytes */
|
||||||
|
orr tmp2, cursor, #0xf
|
||||||
|
add tmp2, tmp2, #1
|
||||||
|
/* If it overflows, fallback to the simple path (unlikely) */
|
||||||
|
cbz tmp2, .Lzeromem_dczva_fallback_entry
|
||||||
|
/*
|
||||||
|
* Next aligned address cannot be after the stop address because the
|
||||||
|
* length cannot be smaller than 16 at this point.
|
||||||
|
*/
|
||||||
|
|
||||||
|
/* First loop: zero byte per byte */
|
||||||
|
1:
|
||||||
|
strb wzr, [cursor], #1
|
||||||
|
cmp cursor, tmp2
|
||||||
|
b.ne 1b
|
||||||
|
.Lzeromem_dczva_initial_1byte_aligned_end:
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Second loop: we need to zero 16 bytes at a time from cursor to tmp1
|
||||||
|
* before being able to use the code that deals with block-size-aligned
|
||||||
|
* addresses.
|
||||||
|
*/
|
||||||
|
cmp cursor, tmp1
|
||||||
|
b.hs 2f
|
||||||
|
1:
|
||||||
|
stp xzr, xzr, [cursor], #16
|
||||||
|
cmp cursor, tmp1
|
||||||
|
b.lo 1b
|
||||||
|
2:
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Third loop: zero a block at a time using DC ZVA cache block zeroing
|
||||||
|
* instruction.
|
||||||
|
*/
|
||||||
|
.Lzeromem_dczva_blocksize_aligned:
|
||||||
|
/*
|
||||||
|
* Calculate the last block-size-aligned address. If the result equals
|
||||||
|
* to the start address, the loop will exit immediately.
|
||||||
|
*/
|
||||||
|
bic tmp1, stop_address, block_mask
|
||||||
|
|
||||||
|
cmp cursor, tmp1
|
||||||
|
b.hs 2f
|
||||||
|
1:
|
||||||
|
/* Zero the block containing the cursor */
|
||||||
|
dc zva, cursor
|
||||||
|
/* Increment the cursor by the size of a block */
|
||||||
|
add cursor, cursor, block_size
|
||||||
|
cmp cursor, tmp1
|
||||||
|
b.lo 1b
|
||||||
|
2:
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Fourth loop: zero 16 bytes at a time and then byte per byte the
|
||||||
|
* remaining area
|
||||||
|
*/
|
||||||
|
.Lzeromem_dczva_final_16bytes_aligned:
|
||||||
|
/*
|
||||||
|
* Calculate the last 16 bytes aligned address. It is assumed that the
|
||||||
|
* block size will never be smaller than 16 bytes so that the current
|
||||||
|
* cursor is aligned to at least 16 bytes boundary.
|
||||||
|
*/
|
||||||
|
bic tmp1, stop_address, #15
|
||||||
|
|
||||||
|
cmp cursor, tmp1
|
||||||
|
b.hs 2f
|
||||||
|
1:
|
||||||
|
stp xzr, xzr, [cursor], #16
|
||||||
|
cmp cursor, tmp1
|
||||||
|
b.lo 1b
|
||||||
|
2:
|
||||||
|
|
||||||
|
/* Fifth and final loop: zero byte per byte */
|
||||||
|
.Lzeromem_dczva_final_1byte_aligned:
|
||||||
|
cmp cursor, stop_address
|
||||||
|
b.eq 2f
|
||||||
|
1:
|
||||||
|
strb wzr, [cursor], #1
|
||||||
|
cmp cursor, stop_address
|
||||||
|
b.ne 1b
|
||||||
|
2:
|
||||||
|
ret
|
||||||
|
|
||||||
|
/* Fallback for unaligned start addresses */
|
||||||
|
.Lzeromem_dczva_fallback_entry:
|
||||||
|
/*
|
||||||
|
* If the start address is already aligned to 16 bytes, skip this loop.
|
||||||
|
*/
|
||||||
|
tst cursor, #0xf
|
||||||
|
b.eq .Lzeromem_dczva_final_16bytes_aligned
|
||||||
|
|
||||||
|
/* Calculate the next address aligned to 16 bytes */
|
||||||
|
orr tmp1, cursor, #15
|
||||||
|
add tmp1, tmp1, #1
|
||||||
|
/* If it overflows, fallback to byte per byte zeroing */
|
||||||
|
cbz tmp1, .Lzeromem_dczva_final_1byte_aligned
|
||||||
|
/* If the next aligned address is after the stop address, fall back */
|
||||||
|
cmp tmp1, stop_address
|
||||||
|
b.hs .Lzeromem_dczva_final_1byte_aligned
|
||||||
|
|
||||||
|
/* Fallback entry loop: zero byte per byte */
|
||||||
|
1:
|
||||||
|
strb wzr, [cursor], #1
|
||||||
|
cmp cursor, tmp1
|
||||||
|
b.ne 1b
|
||||||
|
|
||||||
|
b .Lzeromem_dczva_final_16bytes_aligned
|
||||||
|
|
||||||
|
.unreq cursor
|
||||||
|
/*
|
||||||
|
* length is already unreq'ed to reuse the register for another
|
||||||
|
* variable.
|
||||||
|
*/
|
||||||
|
.unreq stop_address
|
||||||
|
.unreq block_size
|
||||||
|
.unreq block_mask
|
||||||
|
.unreq tmp1
|
||||||
|
.unreq tmp2
|
||||||
|
endfunc zeromem_dczva
|
||||||
|
|
||||||
/* --------------------------------------------------------------------------
|
/* --------------------------------------------------------------------------
|
||||||
* void memcpy16(void *dest, const void *src, unsigned int length)
|
* void memcpy16(void *dest, const void *src, unsigned int length)
|
||||||
|
|
|
@ -1,5 +1,5 @@
|
||||||
/*
|
/*
|
||||||
* Copyright (c) 2016, ARM Limited and Contributors. All rights reserved.
|
* Copyright (c) 2016-2017, ARM Limited and Contributors. All rights reserved.
|
||||||
*
|
*
|
||||||
* Redistribution and use in source and binary forms, with or without
|
* Redistribution and use in source and binary forms, with or without
|
||||||
* modification, are permitted provided that the following conditions are met:
|
* modification, are permitted provided that the following conditions are met:
|
||||||
|
@ -38,6 +38,7 @@
|
||||||
#include <platform_def.h>
|
#include <platform_def.h>
|
||||||
#include <smcc_helpers.h>
|
#include <smcc_helpers.h>
|
||||||
#include <string.h>
|
#include <string.h>
|
||||||
|
#include <utils.h>
|
||||||
|
|
||||||
/*******************************************************************************
|
/*******************************************************************************
|
||||||
* Context management library initialisation routine. This library is used by
|
* Context management library initialisation routine. This library is used by
|
||||||
|
@ -84,7 +85,7 @@ static void cm_init_context_common(cpu_context_t *ctx, const entry_point_info_t
|
||||||
security_state = GET_SECURITY_STATE(ep->h.attr);
|
security_state = GET_SECURITY_STATE(ep->h.attr);
|
||||||
|
|
||||||
/* Clear any residual register values from the context */
|
/* Clear any residual register values from the context */
|
||||||
memset(ctx, 0, sizeof(*ctx));
|
zeromem(ctx, sizeof(*ctx));
|
||||||
|
|
||||||
reg_ctx = get_regs_ctx(ctx);
|
reg_ctx = get_regs_ctx(ctx);
|
||||||
|
|
||||||
|
|
|
@ -1,5 +1,5 @@
|
||||||
/*
|
/*
|
||||||
* Copyright (c) 2013-2016, ARM Limited and Contributors. All rights reserved.
|
* Copyright (c) 2013-2017, ARM Limited and Contributors. All rights reserved.
|
||||||
*
|
*
|
||||||
* Redistribution and use in source and binary forms, with or without
|
* Redistribution and use in source and binary forms, with or without
|
||||||
* modification, are permitted provided that the following conditions are met:
|
* modification, are permitted provided that the following conditions are met:
|
||||||
|
@ -39,6 +39,7 @@
|
||||||
#include <platform_def.h>
|
#include <platform_def.h>
|
||||||
#include <smcc_helpers.h>
|
#include <smcc_helpers.h>
|
||||||
#include <string.h>
|
#include <string.h>
|
||||||
|
#include <utils.h>
|
||||||
|
|
||||||
|
|
||||||
/*******************************************************************************
|
/*******************************************************************************
|
||||||
|
@ -91,7 +92,7 @@ static void cm_init_context_common(cpu_context_t *ctx, const entry_point_info_t
|
||||||
security_state = GET_SECURITY_STATE(ep->h.attr);
|
security_state = GET_SECURITY_STATE(ep->h.attr);
|
||||||
|
|
||||||
/* Clear any residual register values from the context */
|
/* Clear any residual register values from the context */
|
||||||
memset(ctx, 0, sizeof(*ctx));
|
zeromem(ctx, sizeof(*ctx));
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* Base the context SCR on the current value, adjust for entry point
|
* Base the context SCR on the current value, adjust for entry point
|
||||||
|
|
|
@ -37,6 +37,7 @@
|
||||||
#include <debug.h>
|
#include <debug.h>
|
||||||
#include <platform.h>
|
#include <platform.h>
|
||||||
#include <string.h>
|
#include <string.h>
|
||||||
|
#include <utils.h>
|
||||||
#include "psci_private.h"
|
#include "psci_private.h"
|
||||||
|
|
||||||
/*
|
/*
|
||||||
|
@ -622,7 +623,7 @@ static int psci_get_ns_ep_info(entry_point_info_t *ep,
|
||||||
SET_PARAM_HEAD(ep, PARAM_EP, VERSION_1, ep_attr);
|
SET_PARAM_HEAD(ep, PARAM_EP, VERSION_1, ep_attr);
|
||||||
|
|
||||||
ep->pc = entrypoint;
|
ep->pc = entrypoint;
|
||||||
memset(&ep->args, 0, sizeof(ep->args));
|
zeromem(&ep->args, sizeof(ep->args));
|
||||||
ep->args.arg0 = context_id;
|
ep->args.arg0 = context_id;
|
||||||
|
|
||||||
mode = scr & SCR_HCE_BIT ? MODE32_hyp : MODE32_svc;
|
mode = scr & SCR_HCE_BIT ? MODE32_hyp : MODE32_svc;
|
||||||
|
@ -659,7 +660,7 @@ static int psci_get_ns_ep_info(entry_point_info_t *ep,
|
||||||
SET_PARAM_HEAD(ep, PARAM_EP, VERSION_1, ep_attr);
|
SET_PARAM_HEAD(ep, PARAM_EP, VERSION_1, ep_attr);
|
||||||
|
|
||||||
ep->pc = entrypoint;
|
ep->pc = entrypoint;
|
||||||
memset(&ep->args, 0, sizeof(ep->args));
|
zeromem(&ep->args, sizeof(ep->args));
|
||||||
ep->args.arg0 = context_id;
|
ep->args.arg0 = context_id;
|
||||||
|
|
||||||
/*
|
/*
|
||||||
|
@ -951,7 +952,7 @@ unsigned int psci_get_max_phys_off_afflvl(void)
|
||||||
{
|
{
|
||||||
psci_power_state_t state_info;
|
psci_power_state_t state_info;
|
||||||
|
|
||||||
memset(&state_info, 0, sizeof(state_info));
|
zeromem(&state_info, sizeof(state_info));
|
||||||
psci_get_target_local_pwr_states(PLAT_MAX_PWR_LVL, &state_info);
|
psci_get_target_local_pwr_states(PLAT_MAX_PWR_LVL, &state_info);
|
||||||
|
|
||||||
return psci_find_target_suspend_lvl(&state_info);
|
return psci_find_target_suspend_lvl(&state_info);
|
||||||
|
|
|
@ -1,5 +1,5 @@
|
||||||
/*
|
/*
|
||||||
* Copyright (c) 2015-2016, ARM Limited and Contributors. All rights reserved.
|
* Copyright (c) 2015-2017, ARM Limited and Contributors. All rights reserved.
|
||||||
*
|
*
|
||||||
* Redistribution and use in source and binary forms, with or without
|
* Redistribution and use in source and binary forms, with or without
|
||||||
* modification, are permitted provided that the following conditions are met:
|
* modification, are permitted provided that the following conditions are met:
|
||||||
|
@ -38,6 +38,7 @@
|
||||||
#include <plat_arm.h>
|
#include <plat_arm.h>
|
||||||
#include <platform_def.h>
|
#include <platform_def.h>
|
||||||
#include <string.h>
|
#include <string.h>
|
||||||
|
#include <utils.h>
|
||||||
|
|
||||||
/* Data structure which holds the extents of the trusted SRAM for BL2 */
|
/* Data structure which holds the extents of the trusted SRAM for BL2 */
|
||||||
static meminfo_t bl2_tzram_layout __aligned(CACHE_WRITEBACK_GRANULE);
|
static meminfo_t bl2_tzram_layout __aligned(CACHE_WRITEBACK_GRANULE);
|
||||||
|
@ -123,7 +124,7 @@ bl31_params_t *bl2_plat_get_bl31_params(void)
|
||||||
* Initialise the memory for all the arguments that needs to
|
* Initialise the memory for all the arguments that needs to
|
||||||
* be passed to BL31
|
* be passed to BL31
|
||||||
*/
|
*/
|
||||||
memset(&bl31_params_mem, 0, sizeof(bl2_to_bl31_params_mem_t));
|
zeromem(&bl31_params_mem, sizeof(bl2_to_bl31_params_mem_t));
|
||||||
|
|
||||||
/* Assign memory for TF related information */
|
/* Assign memory for TF related information */
|
||||||
bl2_to_bl31_params = &bl31_params_mem.bl31_params;
|
bl2_to_bl31_params = &bl31_params_mem.bl31_params;
|
||||||
|
|
|
@ -1,5 +1,5 @@
|
||||||
/*
|
/*
|
||||||
* Copyright (c) 2015, ARM Limited and Contributors. All rights reserved.
|
* Copyright (c) 2015-2017, ARM Limited and Contributors. All rights reserved.
|
||||||
*
|
*
|
||||||
* Redistribution and use in source and binary forms, with or without
|
* Redistribution and use in source and binary forms, with or without
|
||||||
* modification, are permitted provided that the following conditions are met:
|
* modification, are permitted provided that the following conditions are met:
|
||||||
|
@ -98,7 +98,7 @@ void bl2_platform_setup(void)
|
||||||
* - restoring the SCP boot configuration.
|
* - restoring the SCP boot configuration.
|
||||||
*/
|
*/
|
||||||
VERBOSE("BL2: Restoring SCP reset data in Trusted SRAM\n");
|
VERBOSE("BL2: Restoring SCP reset data in Trusted SRAM\n");
|
||||||
memset((void *) ARM_TRUSTED_SRAM_BASE, 0, 128);
|
zero_normalmem((void *)ARM_TRUSTED_SRAM_BASE, 128);
|
||||||
mmio_write_32(SCP_BOOT_CFG_ADDR, scp_boot_config);
|
mmio_write_32(SCP_BOOT_CFG_ADDR, scp_boot_config);
|
||||||
}
|
}
|
||||||
#endif /* EL3_PAYLOAD_BASE */
|
#endif /* EL3_PAYLOAD_BASE */
|
||||||
|
|
|
@ -1,5 +1,5 @@
|
||||||
/*
|
/*
|
||||||
* Copyright (c) 2014-2016, ARM Limited and Contributors. All rights reserved.
|
* Copyright (c) 2014-2017, ARM Limited and Contributors. All rights reserved.
|
||||||
*
|
*
|
||||||
* Redistribution and use in source and binary forms, with or without
|
* Redistribution and use in source and binary forms, with or without
|
||||||
* modification, are permitted provided that the following conditions are met:
|
* modification, are permitted provided that the following conditions are met:
|
||||||
|
@ -34,6 +34,7 @@
|
||||||
#include <debug.h>
|
#include <debug.h>
|
||||||
#include <platform.h>
|
#include <platform.h>
|
||||||
#include <string.h>
|
#include <string.h>
|
||||||
|
#include <utils.h>
|
||||||
#include "css_mhu.h"
|
#include "css_mhu.h"
|
||||||
#include "css_scpi.h"
|
#include "css_scpi.h"
|
||||||
|
|
||||||
|
@ -204,7 +205,8 @@ int scpi_get_css_power_state(unsigned int mpidr, unsigned int *cpu_state_p,
|
||||||
scpi_secure_message_start();
|
scpi_secure_message_start();
|
||||||
|
|
||||||
/* Populate request headers */
|
/* Populate request headers */
|
||||||
cmd = memset(SCPI_CMD_HEADER_AP_TO_SCP, 0, sizeof(*cmd));
|
zeromem(SCPI_CMD_HEADER_AP_TO_SCP, sizeof(*cmd));
|
||||||
|
cmd = SCPI_CMD_HEADER_AP_TO_SCP;
|
||||||
cmd->id = SCPI_CMD_GET_CSS_POWER_STATE;
|
cmd->id = SCPI_CMD_GET_CSS_POWER_STATE;
|
||||||
|
|
||||||
/*
|
/*
|
||||||
|
|
|
@ -1,5 +1,5 @@
|
||||||
/*
|
/*
|
||||||
* Copyright (c) 2016, ARM Limited and Contributors. All rights reserved.
|
* Copyright (c) 2016-2017, ARM Limited and Contributors. All rights reserved.
|
||||||
*
|
*
|
||||||
* Redistribution and use in source and binary forms, with or without
|
* Redistribution and use in source and binary forms, with or without
|
||||||
* modification, are permitted provided that the following conditions are met:
|
* modification, are permitted provided that the following conditions are met:
|
||||||
|
@ -113,7 +113,8 @@ SECTIONS
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* The .bss section gets initialised to 0 at runtime.
|
* The .bss section gets initialised to 0 at runtime.
|
||||||
* Its base address must be 16-byte aligned.
|
* Its base address should be 16-byte aligned for better performance of the
|
||||||
|
* zero-initialization code.
|
||||||
*/
|
*/
|
||||||
.bss (NOLOAD) : ALIGN(16) {
|
.bss (NOLOAD) : ALIGN(16) {
|
||||||
__BSS_START__ = .;
|
__BSS_START__ = .;
|
||||||
|
|
|
@ -1,5 +1,5 @@
|
||||||
/*
|
/*
|
||||||
* Copyright (c) 2015, ARM Limited and Contributors. All rights reserved.
|
* Copyright (c) 2015-2017, ARM Limited and Contributors. All rights reserved.
|
||||||
*
|
*
|
||||||
* Redistribution and use in source and binary forms, with or without
|
* Redistribution and use in source and binary forms, with or without
|
||||||
* modification, are permitted provided that the following conditions are met:
|
* modification, are permitted provided that the following conditions are met:
|
||||||
|
@ -35,10 +35,9 @@
|
||||||
#include <memctrl.h>
|
#include <memctrl.h>
|
||||||
#include <string.h>
|
#include <string.h>
|
||||||
#include <tegra_def.h>
|
#include <tegra_def.h>
|
||||||
|
#include <utils.h>
|
||||||
#include <xlat_tables.h>
|
#include <xlat_tables.h>
|
||||||
|
|
||||||
extern void zeromem16(void *mem, unsigned int length);
|
|
||||||
|
|
||||||
#define TEGRA_GPU_RESET_REG_OFFSET 0x28c
|
#define TEGRA_GPU_RESET_REG_OFFSET 0x28c
|
||||||
#define GPU_RESET_BIT (1 << 24)
|
#define GPU_RESET_BIT (1 << 24)
|
||||||
|
|
||||||
|
@ -114,13 +113,13 @@ static void tegra_clear_videomem(uintptr_t non_overlap_area_start,
|
||||||
* Perform cache maintenance to ensure that the non-overlapping area is
|
* Perform cache maintenance to ensure that the non-overlapping area is
|
||||||
* zeroed out. The first invalidation of this range ensures that
|
* zeroed out. The first invalidation of this range ensures that
|
||||||
* possible evictions of dirty cache lines do not interfere with the
|
* possible evictions of dirty cache lines do not interfere with the
|
||||||
* 'zeromem16' operation. Other CPUs could speculatively prefetch the
|
* 'zeromem' operation. Other CPUs could speculatively prefetch the
|
||||||
* main memory contents of this area between the first invalidation and
|
* main memory contents of this area between the first invalidation and
|
||||||
* the 'zeromem16' operation. The second invalidation ensures that any
|
* the 'zeromem' operation. The second invalidation ensures that any
|
||||||
* such cache lines are removed as well.
|
* such cache lines are removed as well.
|
||||||
*/
|
*/
|
||||||
inv_dcache_range(non_overlap_area_start, non_overlap_area_size);
|
inv_dcache_range(non_overlap_area_start, non_overlap_area_size);
|
||||||
zeromem16((void *)non_overlap_area_start, non_overlap_area_size);
|
zeromem((void *)non_overlap_area_start, non_overlap_area_size);
|
||||||
inv_dcache_range(non_overlap_area_start, non_overlap_area_size);
|
inv_dcache_range(non_overlap_area_start, non_overlap_area_size);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -1,5 +1,5 @@
|
||||||
/*
|
/*
|
||||||
* Copyright (c) 2015-2016, ARM Limited and Contributors. All rights reserved.
|
* Copyright (c) 2015-2017, ARM Limited and Contributors. All rights reserved.
|
||||||
*
|
*
|
||||||
* Redistribution and use in source and binary forms, with or without
|
* Redistribution and use in source and binary forms, with or without
|
||||||
* modification, are permitted provided that the following conditions are met:
|
* modification, are permitted provided that the following conditions are met:
|
||||||
|
@ -35,7 +35,7 @@
|
||||||
#include <platform_def.h>
|
#include <platform_def.h>
|
||||||
#include "qemu_private.h"
|
#include "qemu_private.h"
|
||||||
#include <string.h>
|
#include <string.h>
|
||||||
|
#include <utils.h>
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* The next 2 constants identify the extents of the code & RO data region.
|
* The next 2 constants identify the extents of the code & RO data region.
|
||||||
|
@ -91,7 +91,7 @@ bl31_params_t *bl2_plat_get_bl31_params(void)
|
||||||
* Initialise the memory for all the arguments that needs to
|
* Initialise the memory for all the arguments that needs to
|
||||||
* be passed to BL3-1
|
* be passed to BL3-1
|
||||||
*/
|
*/
|
||||||
memset(&bl31_params_mem, 0, sizeof(bl2_to_bl31_params_mem_t));
|
zeromem(&bl31_params_mem, sizeof(bl2_to_bl31_params_mem_t));
|
||||||
|
|
||||||
/* Assign memory for TF related information */
|
/* Assign memory for TF related information */
|
||||||
bl2_to_bl31_params = &bl31_params_mem.bl31_params;
|
bl2_to_bl31_params = &bl31_params_mem.bl31_params;
|
||||||
|
|
|
@ -1,5 +1,5 @@
|
||||||
/*
|
/*
|
||||||
* Copyright (c) 2016, ARM Limited and Contributors. All rights reserved.
|
* Copyright (c) 2016-2017, ARM Limited and Contributors. All rights reserved.
|
||||||
*
|
*
|
||||||
* Redistribution and use in source and binary forms, with or without
|
* Redistribution and use in source and binary forms, with or without
|
||||||
* modification, are permitted provided that the following conditions are met:
|
* modification, are permitted provided that the following conditions are met:
|
||||||
|
@ -31,6 +31,7 @@
|
||||||
#include <string.h>
|
#include <string.h>
|
||||||
#include <stdint.h>
|
#include <stdint.h>
|
||||||
#include <dram.h>
|
#include <dram.h>
|
||||||
|
#include <utils.h>
|
||||||
#include "dram_spec_timing.h"
|
#include "dram_spec_timing.h"
|
||||||
|
|
||||||
static const uint8_t ddr3_cl_cwl[][7] = {
|
static const uint8_t ddr3_cl_cwl[][7] = {
|
||||||
|
@ -228,7 +229,7 @@ static void ddr3_get_parameter(struct timing_related_config *timing_config,
|
||||||
uint32_t ddr_capability_per_die = get_max_die_capability(timing_config);
|
uint32_t ddr_capability_per_die = get_max_die_capability(timing_config);
|
||||||
uint32_t tmp;
|
uint32_t tmp;
|
||||||
|
|
||||||
memset((void *)pdram_timing, 0, sizeof(struct dram_timing_t));
|
zeromem((void *)pdram_timing, sizeof(struct dram_timing_t));
|
||||||
pdram_timing->mhz = nmhz;
|
pdram_timing->mhz = nmhz;
|
||||||
pdram_timing->al = 0;
|
pdram_timing->al = 0;
|
||||||
pdram_timing->bl = timing_config->bl;
|
pdram_timing->bl = timing_config->bl;
|
||||||
|
@ -441,7 +442,7 @@ static void lpddr2_get_parameter(struct timing_related_config *timing_config,
|
||||||
uint32_t ddr_capability_per_die = get_max_die_capability(timing_config);
|
uint32_t ddr_capability_per_die = get_max_die_capability(timing_config);
|
||||||
uint32_t tmp, trp_tmp, trppb_tmp, tras_tmp, twr_tmp, bl_tmp;
|
uint32_t tmp, trp_tmp, trppb_tmp, tras_tmp, twr_tmp, bl_tmp;
|
||||||
|
|
||||||
memset((void *)pdram_timing, 0, sizeof(struct dram_timing_t));
|
zeromem((void *)pdram_timing, sizeof(struct dram_timing_t));
|
||||||
pdram_timing->mhz = nmhz;
|
pdram_timing->mhz = nmhz;
|
||||||
pdram_timing->al = 0;
|
pdram_timing->al = 0;
|
||||||
pdram_timing->bl = timing_config->bl;
|
pdram_timing->bl = timing_config->bl;
|
||||||
|
@ -678,7 +679,7 @@ static void lpddr3_get_parameter(struct timing_related_config *timing_config,
|
||||||
uint32_t ddr_capability_per_die = get_max_die_capability(timing_config);
|
uint32_t ddr_capability_per_die = get_max_die_capability(timing_config);
|
||||||
uint32_t tmp, trp_tmp, trppb_tmp, tras_tmp, twr_tmp, bl_tmp;
|
uint32_t tmp, trp_tmp, trppb_tmp, tras_tmp, twr_tmp, bl_tmp;
|
||||||
|
|
||||||
memset((void *)pdram_timing, 0, sizeof(struct dram_timing_t));
|
zeromem((void *)pdram_timing, sizeof(struct dram_timing_t));
|
||||||
pdram_timing->mhz = nmhz;
|
pdram_timing->mhz = nmhz;
|
||||||
pdram_timing->al = 0;
|
pdram_timing->al = 0;
|
||||||
pdram_timing->bl = timing_config->bl;
|
pdram_timing->bl = timing_config->bl;
|
||||||
|
@ -968,7 +969,7 @@ static void lpddr4_get_parameter(struct timing_related_config *timing_config,
|
||||||
uint32_t ddr_capability_per_die = get_max_die_capability(timing_config);
|
uint32_t ddr_capability_per_die = get_max_die_capability(timing_config);
|
||||||
uint32_t tmp, trp_tmp, trppb_tmp, tras_tmp;
|
uint32_t tmp, trp_tmp, trppb_tmp, tras_tmp;
|
||||||
|
|
||||||
memset((void *)pdram_timing, 0, sizeof(struct dram_timing_t));
|
zeromem((void *)pdram_timing, sizeof(struct dram_timing_t));
|
||||||
pdram_timing->mhz = nmhz;
|
pdram_timing->mhz = nmhz;
|
||||||
pdram_timing->al = 0;
|
pdram_timing->al = 0;
|
||||||
pdram_timing->bl = timing_config->bl;
|
pdram_timing->bl = timing_config->bl;
|
||||||
|
|
|
@ -1,5 +1,5 @@
|
||||||
/*
|
/*
|
||||||
* Copyright (c) 2013-2016, ARM Limited and Contributors. All rights reserved.
|
* Copyright (c) 2013-2017, ARM Limited and Contributors. All rights reserved.
|
||||||
*
|
*
|
||||||
* Redistribution and use in source and binary forms, with or without
|
* Redistribution and use in source and binary forms, with or without
|
||||||
* modification, are permitted provided that the following conditions are met:
|
* modification, are permitted provided that the following conditions are met:
|
||||||
|
@ -40,6 +40,7 @@
|
||||||
#include <bl_common.h>
|
#include <bl_common.h>
|
||||||
#include <mmio.h>
|
#include <mmio.h>
|
||||||
#include <string.h>
|
#include <string.h>
|
||||||
|
#include <utils.h>
|
||||||
#include "pm_api_sys.h"
|
#include "pm_api_sys.h"
|
||||||
#include "pm_client.h"
|
#include "pm_client.h"
|
||||||
#include "pm_ipi.h"
|
#include "pm_ipi.h"
|
||||||
|
@ -188,7 +189,7 @@ static void pm_client_set_wakeup_sources(void)
|
||||||
uint8_t pm_wakeup_nodes_set[NODE_MAX];
|
uint8_t pm_wakeup_nodes_set[NODE_MAX];
|
||||||
uintptr_t isenabler1 = BASE_GICD_BASE + GICD_ISENABLER + 4;
|
uintptr_t isenabler1 = BASE_GICD_BASE + GICD_ISENABLER + 4;
|
||||||
|
|
||||||
memset(&pm_wakeup_nodes_set, 0, sizeof(pm_wakeup_nodes_set));
|
zeromem(&pm_wakeup_nodes_set, sizeof(pm_wakeup_nodes_set));
|
||||||
|
|
||||||
for (reg_num = 0; reg_num < NUM_GICD_ISENABLER; reg_num++) {
|
for (reg_num = 0; reg_num < NUM_GICD_ISENABLER; reg_num++) {
|
||||||
uint32_t base_irq = reg_num << ISENABLER_SHIFT;
|
uint32_t base_irq = reg_num << ISENABLER_SHIFT;
|
||||||
|
|
|
@ -1,5 +1,5 @@
|
||||||
/*
|
/*
|
||||||
* Copyright (c) 2013-2014, ARM Limited and Contributors. All rights reserved.
|
* Copyright (c) 2013-2017, ARM Limited and Contributors. All rights reserved.
|
||||||
*
|
*
|
||||||
* Redistribution and use in source and binary forms, with or without
|
* Redistribution and use in source and binary forms, with or without
|
||||||
* modification, are permitted provided that the following conditions are met:
|
* modification, are permitted provided that the following conditions are met:
|
||||||
|
@ -33,6 +33,7 @@
|
||||||
#include <bl_common.h>
|
#include <bl_common.h>
|
||||||
#include <context_mgmt.h>
|
#include <context_mgmt.h>
|
||||||
#include <string.h>
|
#include <string.h>
|
||||||
|
#include <utils.h>
|
||||||
#include "opteed_private.h"
|
#include "opteed_private.h"
|
||||||
|
|
||||||
/*******************************************************************************
|
/*******************************************************************************
|
||||||
|
@ -73,7 +74,7 @@ void opteed_init_optee_ep_state(struct entry_point_info *optee_entry_point,
|
||||||
DAIF_FIQ_BIT |
|
DAIF_FIQ_BIT |
|
||||||
DAIF_IRQ_BIT |
|
DAIF_IRQ_BIT |
|
||||||
DAIF_ABT_BIT);
|
DAIF_ABT_BIT);
|
||||||
memset(&optee_entry_point->args, 0, sizeof(optee_entry_point->args));
|
zeromem(&optee_entry_point->args, sizeof(optee_entry_point->args));
|
||||||
}
|
}
|
||||||
|
|
||||||
/*******************************************************************************
|
/*******************************************************************************
|
||||||
|
|
|
@ -1,5 +1,5 @@
|
||||||
/*
|
/*
|
||||||
* Copyright (c) 2013-2016, ARM Limited and Contributors. All rights reserved.
|
* Copyright (c) 2013-2017, ARM Limited and Contributors. All rights reserved.
|
||||||
*
|
*
|
||||||
* Redistribution and use in source and binary forms, with or without
|
* Redistribution and use in source and binary forms, with or without
|
||||||
* modification, are permitted provided that the following conditions are met:
|
* modification, are permitted provided that the following conditions are met:
|
||||||
|
@ -35,6 +35,7 @@
|
||||||
#include <debug.h>
|
#include <debug.h>
|
||||||
#include <string.h>
|
#include <string.h>
|
||||||
#include <tsp.h>
|
#include <tsp.h>
|
||||||
|
#include <utils.h>
|
||||||
#include "tspd_private.h"
|
#include "tspd_private.h"
|
||||||
|
|
||||||
/*******************************************************************************
|
/*******************************************************************************
|
||||||
|
@ -78,7 +79,7 @@ void tspd_init_tsp_ep_state(struct entry_point_info *tsp_entry_point,
|
||||||
tsp_entry_point->spsr = SPSR_64(MODE_EL1,
|
tsp_entry_point->spsr = SPSR_64(MODE_EL1,
|
||||||
MODE_SP_ELX,
|
MODE_SP_ELX,
|
||||||
DISABLE_ALL_EXCEPTIONS);
|
DISABLE_ALL_EXCEPTIONS);
|
||||||
memset(&tsp_entry_point->args, 0, sizeof(tsp_entry_point->args));
|
zeromem(&tsp_entry_point->args, sizeof(tsp_entry_point->args));
|
||||||
}
|
}
|
||||||
|
|
||||||
/*******************************************************************************
|
/*******************************************************************************
|
||||||
|
|
Loading…
Add table
Reference in a new issue