2019-06-30 04:03:00 +03:00
|
|
|
/*
|
|
|
|
* BPMP-Lite Cache/MMU and Frequency driver for Tegra X1
|
|
|
|
*
|
2020-03-14 09:24:24 +02:00
|
|
|
* Copyright (c) 2019-2020 CTCaer
|
2019-06-30 04:03:00 +03:00
|
|
|
*
|
|
|
|
* This program is free software; you can redistribute it and/or modify it
|
|
|
|
* under the terms and conditions of the GNU General Public License,
|
|
|
|
* version 2, as published by the Free Software Foundation.
|
|
|
|
*
|
|
|
|
* This program is distributed in the hope it will be useful, but WITHOUT
|
|
|
|
* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
|
|
|
|
* FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for
|
|
|
|
* more details.
|
|
|
|
*
|
|
|
|
* You should have received a copy of the GNU General Public License
|
|
|
|
* along with this program. If not, see <http://www.gnu.org/licenses/>.
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include "bpmp.h"
|
|
|
|
#include "clock.h"
|
|
|
|
#include "t210.h"
|
2019-12-08 02:23:03 +02:00
|
|
|
#include "../../../common/memory_map.h"
|
2019-06-30 04:03:00 +03:00
|
|
|
#include "../utils/util.h"
|
|
|
|
|
2019-12-14 22:21:42 +02:00
|
|
|
#define BPMP_MMU_CACHE_LINE_SIZE 0x20
|
|
|
|
|
2019-06-30 04:03:00 +03:00
|
|
|
#define BPMP_CACHE_CONFIG 0x0
|
2019-12-14 22:21:42 +02:00
|
|
|
#define CFG_ENABLE_CACHE (1 << 0)
|
|
|
|
#define CFG_ENABLE_SKEW_ASSOC (1 << 1)
|
|
|
|
#define CFG_DISABLE_RANDOM_ALLOC (1 << 2)
|
2019-06-30 04:03:00 +03:00
|
|
|
#define CFG_FORCE_WRITE_THROUGH (1 << 3)
|
2019-12-14 22:21:42 +02:00
|
|
|
#define CFG_NEVER_ALLOCATE (1 << 6)
|
|
|
|
#define CFG_ENABLE_INTERRUPT (1 << 7)
|
|
|
|
#define CFG_MMU_TAG_MODE(x) (x << 8)
|
|
|
|
#define TAG_MODE_PARALLEL 0
|
|
|
|
#define TAG_MODE_TAG_FIRST 1
|
|
|
|
#define TAG_MODE_MMU_FIRST 2
|
2019-06-30 04:03:00 +03:00
|
|
|
#define CFG_DISABLE_WRITE_BUFFER (1 << 10)
|
|
|
|
#define CFG_DISABLE_READ_BUFFER (1 << 11)
|
2019-12-14 22:21:42 +02:00
|
|
|
#define CFG_ENABLE_HANG_DETECT (1 << 12)
|
2019-06-30 04:03:00 +03:00
|
|
|
#define CFG_FULL_LINE_DIRTY (1 << 13)
|
|
|
|
#define CFG_TAG_CHK_ABRT_ON_ERR (1 << 14)
|
2019-12-14 22:21:42 +02:00
|
|
|
#define CFG_TAG_CHK_CLR_ERR (1 << 15)
|
|
|
|
#define CFG_DISABLE_SAMELINE (1 << 16)
|
|
|
|
#define CFG_OBS_BUS_EN (1 << 31)
|
|
|
|
|
2019-06-30 04:03:00 +03:00
|
|
|
#define BPMP_CACHE_LOCK 0x4
|
2019-12-14 22:21:42 +02:00
|
|
|
#define LOCK_LINE(x) (1 << x)
|
|
|
|
|
2019-06-30 04:03:00 +03:00
|
|
|
#define BPMP_CACHE_SIZE 0xC
|
|
|
|
#define BPMP_CACHE_LFSR 0x10
|
2019-12-14 22:21:42 +02:00
|
|
|
|
2019-06-30 04:03:00 +03:00
|
|
|
#define BPMP_CACHE_TAG_STATUS 0x14
|
2019-12-14 22:21:42 +02:00
|
|
|
#define TAG_STATUS_TAG_CHECK_ERROR (1 << 0)
|
|
|
|
#define TAG_STATUS_CONFLICT_ADDR_MASK 0xFFFFFFE0
|
|
|
|
|
2019-06-30 04:03:00 +03:00
|
|
|
#define BPMP_CACHE_CLKEN_OVERRIDE 0x18
|
2019-12-14 22:21:42 +02:00
|
|
|
#define CLKEN_OVERRIDE_WR_MCCIF_CLKEN (1 << 0)
|
|
|
|
#define CLKEN_OVERRIDE_RD_MCCIF_CLKEN (1 << 1)
|
|
|
|
|
2019-06-30 04:03:00 +03:00
|
|
|
#define BPMP_CACHE_MAINT_ADDR 0x20
|
|
|
|
#define BPMP_CACHE_MAINT_DATA 0x24
|
2019-12-14 22:21:42 +02:00
|
|
|
|
2019-06-30 04:03:00 +03:00
|
|
|
#define BPMP_CACHE_MAINT_REQ 0x28
|
|
|
|
#define MAINT_REQ_WAY_BITMAP(x) ((x) << 8)
|
|
|
|
|
|
|
|
#define BPMP_CACHE_INT_MASK 0x40
|
|
|
|
#define BPMP_CACHE_INT_CLEAR 0x44
|
|
|
|
#define BPMP_CACHE_INT_RAW_EVENT 0x48
|
|
|
|
#define BPMP_CACHE_INT_STATUS 0x4C
|
2019-12-14 22:21:42 +02:00
|
|
|
#define INT_MAINT_DONE (1 << 0)
|
|
|
|
#define INT_MAINT_ERROR (1 << 1)
|
2019-06-30 04:03:00 +03:00
|
|
|
|
|
|
|
#define BPMP_CACHE_RB_CFG 0x80
|
|
|
|
#define BPMP_CACHE_WB_CFG 0x84
|
|
|
|
|
|
|
|
#define BPMP_CACHE_MMU_FALLBACK_ENTRY 0xA0
|
|
|
|
#define BPMP_CACHE_MMU_SHADOW_COPY_MASK 0xA4
|
2019-12-14 22:21:42 +02:00
|
|
|
|
2019-06-30 04:03:00 +03:00
|
|
|
#define BPMP_CACHE_MMU_CFG 0xAC
|
2019-12-14 22:21:42 +02:00
|
|
|
#define MMU_CFG_BLOCK_MAIN_ENTRY_WR (1 << 0)
|
2019-06-30 04:03:00 +03:00
|
|
|
#define MMU_CFG_SEQ_EN (1 << 1)
|
|
|
|
#define MMU_CFG_TLB_EN (1 << 2)
|
2019-12-14 22:21:42 +02:00
|
|
|
#define MMU_CFG_SEG_CHECK_ALL_ENTRIES (1 << 3)
|
2019-06-30 04:03:00 +03:00
|
|
|
#define MMU_CFG_ABORT_STORE_LAST (1 << 4)
|
2019-12-14 22:21:42 +02:00
|
|
|
#define MMU_CFG_CLR_ABORT (1 << 5)
|
|
|
|
|
2019-06-30 04:03:00 +03:00
|
|
|
#define BPMP_CACHE_MMU_CMD 0xB0
|
|
|
|
#define MMU_CMD_NOP 0
|
|
|
|
#define MMU_CMD_INIT 1
|
|
|
|
#define MMU_CMD_COPY_SHADOW 2
|
2019-12-14 22:21:42 +02:00
|
|
|
|
2019-06-30 04:03:00 +03:00
|
|
|
#define BPMP_CACHE_MMU_ABORT_STAT 0xB4
|
2019-12-14 22:21:42 +02:00
|
|
|
#define ABORT_STAT_UNIT_MASK 0x7
|
|
|
|
#define ABORT_STAT_UNIT_NONE 0
|
|
|
|
#define ABORT_STAT_UNIT_CACHE 1
|
|
|
|
#define ABORT_STAT_UNIT_SEQ 2
|
|
|
|
#define ABORT_STAT_UNIT_TLB 3
|
|
|
|
#define ABORT_STAT_UNIT_SEG 4
|
|
|
|
#define ABORT_STAT_UNIT_FALLBACK 5
|
|
|
|
#define ABORT_STAT_OVERLAP (1 << 3)
|
|
|
|
#define ABORT_STAT_ENTRY (0x1F << 4)
|
|
|
|
#define ABORT_STAT_TYPE_MASK (3 << 16)
|
|
|
|
#define ABORT_STAT_TYPE_EXE (0 << 16)
|
|
|
|
#define ABORT_STAT_TYPE_RD (1 << 16)
|
|
|
|
#define ABORT_STAT_TYPE_WR (2 << 16)
|
|
|
|
#define ABORT_STAT_SIZE (3 << 18)
|
|
|
|
#define ABORT_STAT_SEQ (1 << 20)
|
|
|
|
#define ABORT_STAT_PROT (1 << 21)
|
|
|
|
|
2019-06-30 04:03:00 +03:00
|
|
|
#define BPMP_CACHE_MMU_ABORT_ADDR 0xB8
|
|
|
|
#define BPMP_CACHE_MMU_ACTIVE_ENTRIES 0xBC
|
|
|
|
|
2019-12-14 22:21:42 +02:00
|
|
|
#define BPMP_MMU_SHADOW_ENTRY_BASE (BPMP_CACHE_BASE + 0x400)
|
|
|
|
#define BPMP_MMU_MAIN_ENTRY_BASE (BPMP_CACHE_BASE + 0x800)
|
|
|
|
#define MMU_EN_CACHED (1 << 0)
|
|
|
|
#define MMU_EN_EXEC (1 << 1)
|
|
|
|
#define MMU_EN_READ (1 << 2)
|
|
|
|
#define MMU_EN_WRITE (1 << 3)
|
2019-06-30 04:03:00 +03:00
|
|
|
|
|
|
|
bpmp_mmu_entry_t mmu_entries[] =
|
|
|
|
{
|
2019-12-08 02:23:03 +02:00
|
|
|
{ DRAM_START, 0xFFFFFFFF, MMU_EN_READ | MMU_EN_WRITE | MMU_EN_EXEC | MMU_EN_CACHED, true },
|
|
|
|
{ IRAM_BASE, 0x4003FFFF, MMU_EN_READ | MMU_EN_WRITE | MMU_EN_EXEC | MMU_EN_CACHED, true }
|
2019-06-30 04:03:00 +03:00
|
|
|
};
|
|
|
|
|
2019-12-07 01:47:44 +02:00
|
|
|
void bpmp_mmu_maintenance(u32 op, bool force)
|
2019-06-30 04:03:00 +03:00
|
|
|
{
|
2019-12-14 22:21:42 +02:00
|
|
|
if (!force && !(BPMP_CACHE_CTRL(BPMP_CACHE_CONFIG) & CFG_ENABLE_CACHE))
|
2019-06-30 04:03:00 +03:00
|
|
|
return;
|
|
|
|
|
2019-12-14 22:21:42 +02:00
|
|
|
BPMP_CACHE_CTRL(BPMP_CACHE_INT_CLEAR) = INT_MAINT_DONE;
|
2019-06-30 04:03:00 +03:00
|
|
|
|
|
|
|
// This is a blocking operation.
|
|
|
|
BPMP_CACHE_CTRL(BPMP_CACHE_MAINT_REQ) = MAINT_REQ_WAY_BITMAP(0xF) | op;
|
|
|
|
|
2019-12-14 22:21:42 +02:00
|
|
|
while(!(BPMP_CACHE_CTRL(BPMP_CACHE_INT_RAW_EVENT) & INT_MAINT_DONE))
|
2019-07-06 22:22:47 +03:00
|
|
|
;
|
2019-06-30 04:03:00 +03:00
|
|
|
|
2019-07-06 22:22:47 +03:00
|
|
|
BPMP_CACHE_CTRL(BPMP_CACHE_INT_CLEAR) = BPMP_CACHE_CTRL(BPMP_CACHE_INT_RAW_EVENT);
|
2019-06-30 04:03:00 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
void bpmp_mmu_set_entry(int idx, bpmp_mmu_entry_t *entry, bool apply)
|
|
|
|
{
|
|
|
|
if (idx > 31)
|
|
|
|
return;
|
|
|
|
|
|
|
|
volatile bpmp_mmu_entry_t *mmu_entry = (bpmp_mmu_entry_t *)(BPMP_MMU_SHADOW_ENTRY_BASE + sizeof(bpmp_mmu_entry_t) * idx);
|
|
|
|
|
|
|
|
if (entry->enable)
|
|
|
|
{
|
2019-12-14 22:21:42 +02:00
|
|
|
mmu_entry->start_addr = ALIGN(entry->start_addr, BPMP_MMU_CACHE_LINE_SIZE);
|
|
|
|
mmu_entry->end_addr = ALIGN_DOWN(entry->end_addr, BPMP_MMU_CACHE_LINE_SIZE);
|
2019-06-30 04:03:00 +03:00
|
|
|
mmu_entry->attr = entry->attr;
|
|
|
|
|
|
|
|
BPMP_CACHE_CTRL(BPMP_CACHE_MMU_SHADOW_COPY_MASK) |= (1 << idx);
|
|
|
|
|
|
|
|
if (apply)
|
|
|
|
BPMP_CACHE_CTRL(BPMP_CACHE_MMU_CMD) = MMU_CMD_COPY_SHADOW;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void bpmp_mmu_enable()
|
|
|
|
{
|
2019-12-14 22:21:42 +02:00
|
|
|
if (BPMP_CACHE_CTRL(BPMP_CACHE_CONFIG) & CFG_ENABLE_CACHE)
|
2019-06-30 04:03:00 +03:00
|
|
|
return;
|
|
|
|
|
|
|
|
// Init BPMP MMU.
|
|
|
|
BPMP_CACHE_CTRL(BPMP_CACHE_MMU_CMD) = MMU_CMD_INIT;
|
|
|
|
BPMP_CACHE_CTRL(BPMP_CACHE_MMU_FALLBACK_ENTRY) = MMU_EN_READ | MMU_EN_WRITE | MMU_EN_EXEC; // RWX for non-defined regions.
|
|
|
|
BPMP_CACHE_CTRL(BPMP_CACHE_MMU_CFG) = MMU_CFG_SEQ_EN | MMU_CFG_TLB_EN | MMU_CFG_ABORT_STORE_LAST;
|
|
|
|
|
|
|
|
// Init BPMP MMU entries.
|
|
|
|
BPMP_CACHE_CTRL(BPMP_CACHE_MMU_SHADOW_COPY_MASK) = 0;
|
|
|
|
for (u32 idx = 0; idx < (sizeof(mmu_entries) / sizeof(bpmp_mmu_entry_t)); idx++)
|
|
|
|
bpmp_mmu_set_entry(idx, &mmu_entries[idx], false);
|
|
|
|
|
|
|
|
BPMP_CACHE_CTRL(BPMP_CACHE_MMU_CMD) = MMU_CMD_COPY_SHADOW;
|
|
|
|
|
|
|
|
// Invalidate cache.
|
2019-12-07 01:47:44 +02:00
|
|
|
bpmp_mmu_maintenance(BPMP_MMU_MAINT_INVALID_WAY, true);
|
2019-06-30 04:03:00 +03:00
|
|
|
|
|
|
|
// Enable cache.
|
2019-12-14 22:21:42 +02:00
|
|
|
BPMP_CACHE_CTRL(BPMP_CACHE_CONFIG) = CFG_ENABLE_CACHE | CFG_FORCE_WRITE_THROUGH |
|
|
|
|
CFG_MMU_TAG_MODE(TAG_MODE_PARALLEL) | CFG_TAG_CHK_ABRT_ON_ERR;
|
2019-06-30 04:03:00 +03:00
|
|
|
|
|
|
|
// HW bug. Invalidate cache again.
|
2019-12-07 01:47:44 +02:00
|
|
|
bpmp_mmu_maintenance(BPMP_MMU_MAINT_INVALID_WAY, false);
|
2019-06-30 04:03:00 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
void bpmp_mmu_disable()
|
|
|
|
{
|
2019-12-14 22:21:42 +02:00
|
|
|
if (!(BPMP_CACHE_CTRL(BPMP_CACHE_CONFIG) & CFG_ENABLE_CACHE))
|
2019-06-30 04:03:00 +03:00
|
|
|
return;
|
|
|
|
|
|
|
|
// Clean and invalidate cache.
|
2019-12-07 01:47:44 +02:00
|
|
|
bpmp_mmu_maintenance(BPMP_MMU_MAINT_CLN_INV_WAY, false);
|
2019-06-30 04:03:00 +03:00
|
|
|
|
2019-09-12 23:08:38 +03:00
|
|
|
// Disable cache.
|
2019-06-30 04:03:00 +03:00
|
|
|
BPMP_CACHE_CTRL(BPMP_CACHE_CONFIG) = 0;
|
|
|
|
}
|
|
|
|
|
2019-12-14 22:21:42 +02:00
|
|
|
// APB clock affects RTC, PWM, MEMFETCH, APE, USB, SOR PWM,
|
|
|
|
// I2C host, DC/DSI/DISP. UART gives extra stress.
|
|
|
|
// 92: 100% success ratio. 93-94: 595-602MHz has 99% success ratio. 95: 608MHz less.
|
2020-01-07 06:26:29 +02:00
|
|
|
const u8 pll_divn[] = {
|
2019-06-30 04:03:00 +03:00
|
|
|
0, // BPMP_CLK_NORMAL: 408MHz 0% - 136MHz APB.
|
2019-12-07 02:01:29 +02:00
|
|
|
85, // BPMP_CLK_HIGH_BOOST: 544MHz 33% - 136MHz APB.
|
|
|
|
90, // BPMP_CLK_SUPER_BOOST: 576MHz 41% - 144MHz APB.
|
|
|
|
92 // BPMP_CLK_HYPER_BOOST: 589MHz 44% - 147MHz APB.
|
|
|
|
// Do not use for public releases!
|
|
|
|
//95 // BPMP_CLK_DEV_BOOST: 608MHz 49% - 152MHz APB.
|
2019-06-30 04:03:00 +03:00
|
|
|
};
|
|
|
|
|
|
|
|
bpmp_freq_t bpmp_clock_set = BPMP_CLK_NORMAL;
|
|
|
|
|
2019-12-14 22:21:42 +02:00
|
|
|
void bpmp_clk_rate_get()
|
|
|
|
{
|
2020-01-07 06:26:29 +02:00
|
|
|
bool clk_src_is_pllp = ((CLOCK(CLK_RST_CONTROLLER_SCLK_BURST_POLICY) >> 4) & 7) == 3;
|
2019-12-14 22:21:42 +02:00
|
|
|
|
|
|
|
if (clk_src_is_pllp)
|
|
|
|
bpmp_clock_set = BPMP_CLK_NORMAL;
|
|
|
|
else
|
|
|
|
{
|
|
|
|
bpmp_clock_set = BPMP_CLK_HIGH_BOOST;
|
|
|
|
|
2020-01-07 06:26:29 +02:00
|
|
|
u8 pll_divn_curr = (CLOCK(CLK_RST_CONTROLLER_PLLC_BASE) >> 10) & 0xFF;
|
|
|
|
for (u32 i = 1; i < sizeof(pll_divn); i++)
|
2019-12-14 22:21:42 +02:00
|
|
|
{
|
2020-01-07 06:26:29 +02:00
|
|
|
if (pll_divn[i] == pll_divn_curr)
|
2019-12-14 22:21:42 +02:00
|
|
|
{
|
|
|
|
bpmp_clock_set = i;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-06-30 04:03:00 +03:00
|
|
|
void bpmp_clk_rate_set(bpmp_freq_t fid)
|
|
|
|
{
|
|
|
|
if (fid > (BPMP_CLK_MAX - 1))
|
|
|
|
fid = BPMP_CLK_MAX - 1;
|
|
|
|
|
|
|
|
if (bpmp_clock_set == fid)
|
|
|
|
return;
|
|
|
|
|
|
|
|
if (fid)
|
|
|
|
{
|
|
|
|
if (bpmp_clock_set)
|
|
|
|
{
|
|
|
|
// Restore to PLLP source during PLLC4 configuration.
|
2019-09-09 16:56:37 +03:00
|
|
|
CLOCK(CLK_RST_CONTROLLER_SCLK_BURST_POLICY) = 0x20003333; // PLLP_OUT.
|
2019-12-14 22:21:42 +02:00
|
|
|
msleep(1); // Wait a bit for clock source change.
|
2019-06-30 04:03:00 +03:00
|
|
|
}
|
|
|
|
|
2020-01-07 06:46:22 +02:00
|
|
|
// Configure and enable PLLC.
|
|
|
|
clock_enable_pllc(pll_divn[fid]);
|
2019-06-30 04:03:00 +03:00
|
|
|
|
2019-12-14 22:21:42 +02:00
|
|
|
// Set SCLK / HCLK / PCLK.
|
|
|
|
CLOCK(CLK_RST_CONTROLLER_CLK_SYSTEM_RATE) = 3; // PCLK = HCLK / (3 + 1). HCLK == SCLK.
|
2020-01-07 06:26:29 +02:00
|
|
|
CLOCK(CLK_RST_CONTROLLER_SCLK_BURST_POLICY) = 0x20003310; // PLLC_OUT1 for active and CLKM for idle.
|
2019-06-30 04:03:00 +03:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2020-01-07 06:26:29 +02:00
|
|
|
CLOCK(CLK_RST_CONTROLLER_SCLK_BURST_POLICY) = 0x20003330; // PLLP_OUT for active and CLKM for idle.
|
2019-12-14 22:21:42 +02:00
|
|
|
msleep(1); // Wait a bit for clock source change.
|
|
|
|
CLOCK(CLK_RST_CONTROLLER_CLK_SYSTEM_RATE) = 2; // PCLK = HCLK / (2 + 1). HCLK == SCLK.
|
2019-06-30 04:03:00 +03:00
|
|
|
|
2020-01-07 06:46:22 +02:00
|
|
|
// Disable PLLC to save power.
|
|
|
|
clock_disable_pllc();
|
2019-06-30 04:03:00 +03:00
|
|
|
}
|
2019-12-14 22:21:42 +02:00
|
|
|
bpmp_clock_set = fid;
|
2019-06-30 04:03:00 +03:00
|
|
|
}
|
|
|
|
|
2019-08-28 01:33:38 +03:00
|
|
|
// The following functions halt BPMP to reduce power while sleeping.
|
|
|
|
// They are not as accurate as RTC at big values but they guarantee time+ delay.
|
|
|
|
void bpmp_usleep(u32 us)
|
|
|
|
{
|
|
|
|
u32 delay;
|
|
|
|
|
|
|
|
// Each iteration takes 1us.
|
|
|
|
while (us)
|
|
|
|
{
|
|
|
|
delay = (us > HALT_COP_MAX_CNT) ? HALT_COP_MAX_CNT : us;
|
|
|
|
us -= delay;
|
|
|
|
|
|
|
|
FLOW_CTLR(FLOW_CTLR_HALT_COP_EVENTS) = HALT_COP_WAIT_EVENT | HALT_COP_USEC | delay;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void bpmp_msleep(u32 ms)
|
|
|
|
{
|
|
|
|
u32 delay;
|
|
|
|
|
|
|
|
// Iteration time is variable. ~200 - 1000us.
|
|
|
|
while (ms)
|
|
|
|
{
|
|
|
|
delay = (ms > HALT_COP_MAX_CNT) ? HALT_COP_MAX_CNT : ms;
|
|
|
|
ms -= delay;
|
|
|
|
|
|
|
|
FLOW_CTLR(FLOW_CTLR_HALT_COP_EVENTS) = HALT_COP_WAIT_EVENT | HALT_COP_MSEC | delay;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void bpmp_halt()
|
|
|
|
{
|
|
|
|
FLOW_CTLR(FLOW_CTLR_HALT_COP_EVENTS) = HALT_COP_WAIT_EVENT | HALT_COP_JTAG;
|
|
|
|
}
|
|
|
|
|