blob: 292ff233701de1bd4c83947bdaf1657af442e0da [file] [log] [blame]
/****************************************************************************
* arch/xtensa/src/esp32/esp32_spicache.c
*
* Licensed to the Apache Software Foundation (ASF) under one or more
* contributor license agreements. See the NOTICE file distributed with
* this work for additional information regarding copyright ownership. The
* ASF licenses this file to you under the Apache License, Version 2.0 (the
* "License"); you may not use this file except in compliance with the
* License. You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS, WITHOUT
* WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the
* License for the specific language governing permissions and limitations
* under the License.
*
****************************************************************************/
/****************************************************************************
* Included Files
****************************************************************************/
#include <nuttx/config.h>
#if defined(CONFIG_ESP32_SPIRAM) || defined(CONFIG_ESP32_SPIFLASH)
#include <stdint.h>
#include <debug.h>
#include <stdio.h>
#include <string.h>
#include <sys/types.h>
#include <errno.h>
#include "xtensa.h"
#include "xtensa_attr.h"
#include "hardware/esp32_soc.h"
#include "hardware/esp32_spi.h"
#include "hardware/esp32_dport.h"
#ifdef CONFIG_ESP32_SPIRAM
#include "esp32_spiram.h"
#endif
#include "esp32_spicache.h"
/****************************************************************************
* Pre-processor Definitions
****************************************************************************/
/****************************************************************************
* Private Types
****************************************************************************/
/****************************************************************************
* Private Data
****************************************************************************/
static uint32_t s_flash_op_cache_state[CONFIG_SMP_NCPUS];
/****************************************************************************
* Private Functions
****************************************************************************/
/****************************************************************************
* Public Functions
****************************************************************************/
/****************************************************************************
* Name: spiflash_disable_cache
****************************************************************************/
void IRAM_ATTR spi_disable_cache(int cpu)
{
const uint32_t cache_mask = 0x3f; /* Caches' bits in CTRL1_REG */
uint32_t regval;
uint32_t ret = 0;
if (cpu == 0)
{
ret |= (getreg32(DPORT_PRO_CACHE_CTRL1_REG) & cache_mask);
while (((getreg32(DPORT_PRO_DCACHE_DBUG0_REG) >>
DPORT_PRO_CACHE_STATE_S) & DPORT_PRO_CACHE_STATE) != 1)
{
;
}
regval = getreg32(DPORT_PRO_CACHE_CTRL_REG);
regval &= ~DPORT_PRO_CACHE_ENABLE_M;
putreg32(regval, DPORT_PRO_CACHE_CTRL_REG);
}
#ifdef CONFIG_SMP
else
{
ret |= (getreg32(DPORT_APP_CACHE_CTRL1_REG) & cache_mask);
while (((getreg32(DPORT_APP_DCACHE_DBUG0_REG) >>
DPORT_APP_CACHE_STATE_S) & DPORT_APP_CACHE_STATE) != 1)
{
;
}
regval = getreg32(DPORT_APP_CACHE_CTRL_REG);
regval &= ~DPORT_APP_CACHE_ENABLE_M;
putreg32(regval, DPORT_APP_CACHE_CTRL_REG);
}
#endif
s_flash_op_cache_state[cpu] = ret;
}
/****************************************************************************
* Name: spiflash_enable_cache
****************************************************************************/
void IRAM_ATTR spi_enable_cache(int cpu)
{
const uint32_t cache_mask = 0x3f; /* Caches' bits in CTRL1_REG */
uint32_t regval;
uint32_t ctrlreg;
uint32_t ctrl1reg;
uint32_t ctrlmask;
if (cpu == 0)
{
ctrlreg = DPORT_PRO_CACHE_CTRL_REG;
ctrl1reg = DPORT_PRO_CACHE_CTRL1_REG;
ctrlmask = DPORT_PRO_CACHE_ENABLE_M;
}
#ifdef CONFIG_SMP
else
{
ctrlreg = DPORT_APP_CACHE_CTRL_REG;
ctrl1reg = DPORT_APP_CACHE_CTRL1_REG;
ctrlmask = DPORT_APP_CACHE_ENABLE_M;
}
#endif
regval = getreg32(ctrlreg);
regval |= ctrlmask;
putreg32(regval, ctrlreg);
regval = getreg32(ctrl1reg);
regval &= ~cache_mask;
regval |= s_flash_op_cache_state[cpu];
putreg32(regval, ctrl1reg);
}
/****************************************************************************
* Name: spi_flash_cache_enabled
*
* Description:
* Check at runtime if flash cache is enabled on both CPUs.
*
* Returned Value:
* True if both CPUs have flash cache enabled, false otherwise.
*
****************************************************************************/
bool spi_flash_cache_enabled(void)
{
bool result = false;
if (((getreg32(DPORT_PRO_CACHE_CTRL_REG) & DPORT_PRO_CACHE_ENABLE) != 0)
#ifdef CONFIG_SMP
&& ((getreg32(DPORT_APP_CACHE_CTRL_REG) & DPORT_APP_CACHE_ENABLE) != 0)
#endif
)
{
result = true;
}
return result;
}
#endif /* CONFIG_ESP32_SPICACHE */