2021-06-03 13:40:09 +02:00
|
|
|
/*
|
2022-06-27 15:24:07 +08:00
|
|
|
* SPDX-FileCopyrightText: 2021-2022 Espressif Systems (Shanghai) CO LTD
|
2021-06-03 13:40:09 +02:00
|
|
|
*
|
|
|
|
* SPDX-License-Identifier: Apache-2.0
|
|
|
|
*/
|
2017-07-20 16:26:35 +08:00
|
|
|
/*
|
|
|
|
This code tests the interaction between PSRAM and SPI flash routines.
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include <esp_types.h>
|
|
|
|
#include <stdio.h>
|
|
|
|
#include "freertos/FreeRTOS.h"
|
|
|
|
#include "freertos/task.h"
|
|
|
|
#include "freertos/semphr.h"
|
|
|
|
#include "freertos/queue.h"
|
|
|
|
#include "unity.h"
|
|
|
|
#include <stdint.h>
|
|
|
|
#include <stdlib.h>
|
|
|
|
#include <stdio.h>
|
|
|
|
#include <string.h>
|
2023-05-15 10:34:07 +08:00
|
|
|
#include "test_utils.h"
|
2017-07-20 16:26:35 +08:00
|
|
|
#include "esp_heap_caps.h"
|
2022-06-27 15:24:07 +08:00
|
|
|
#include "spi_flash_mmap.h"
|
2017-07-20 16:26:35 +08:00
|
|
|
#include "esp_partition.h"
|
2019-06-19 15:31:47 +08:00
|
|
|
#include "soc/soc.h"
|
2017-07-20 16:26:35 +08:00
|
|
|
|
2019-06-05 14:34:19 +10:00
|
|
|
#if CONFIG_SPIRAM
|
2017-07-20 16:26:35 +08:00
|
|
|
|
2018-11-19 22:20:52 +05:30
|
|
|
#if CONFIG_SPIRAM_USE_CAPS_ALLOC || CONFIG_SPIRAM_USE_MALLOC
|
|
|
|
#define USE_CAPS_ALLOC 1
|
|
|
|
#endif // CONFIG_SPIRAM_USE_CAPS_ALLOC || CONFIG_SPIRAM_USE_MALLOC
|
2017-07-20 16:26:35 +08:00
|
|
|
|
|
|
|
#define TSTSZ (16*1024)
|
|
|
|
|
2018-11-19 22:20:52 +05:30
|
|
|
#if !CONFIG_FREERTOS_UNICORE
|
2017-07-20 16:26:35 +08:00
|
|
|
|
|
|
|
volatile static int res[2], err[2];
|
|
|
|
|
|
|
|
void tstMem(void *arg) {
|
|
|
|
volatile unsigned char *mem=(volatile unsigned char*)arg;
|
|
|
|
int p=0;
|
|
|
|
while(1) {
|
|
|
|
for (int i=0; i<TSTSZ; i++) {
|
|
|
|
mem[i]=(i^p);
|
|
|
|
}
|
|
|
|
for (int i=0; i<TSTSZ; i++) {
|
|
|
|
if (mem[i]!=((i^p)&0xff)) {
|
|
|
|
printf("Core %d mem err! Got %x espected %x at addr %p\n", xPortGetCoreID(), mem[i], (i^p)&0xff, &mem[i]);
|
|
|
|
err[xPortGetCoreID()]++;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
p++;
|
|
|
|
res[xPortGetCoreID()]++;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2018-11-19 22:20:52 +05:30
|
|
|
TEST_CASE("Spiram cache flush on mmap", "[spiram]")
|
2017-07-20 16:26:35 +08:00
|
|
|
{
|
|
|
|
void *mem[2];
|
|
|
|
res[0]=0; res[1]=0;
|
2018-11-19 22:20:52 +05:30
|
|
|
#if USE_CAPS_ALLOC
|
2017-09-06 16:13:40 +08:00
|
|
|
printf("Allocating SPI RAM chunk...\n");
|
2017-09-05 17:29:57 +08:00
|
|
|
mem[0]=heap_caps_malloc(TSTSZ, MALLOC_CAP_SPIRAM);
|
|
|
|
mem[1]=heap_caps_malloc(TSTSZ, MALLOC_CAP_SPIRAM);
|
2017-07-20 16:26:35 +08:00
|
|
|
#else
|
2019-04-08 18:02:05 +08:00
|
|
|
mem[0]=(void*)SOC_EXTRAM_DATA_LOW;
|
|
|
|
mem[1]=(void*)SOC_EXTRAM_DATA_LOW+TSTSZ;
|
2017-07-20 16:26:35 +08:00
|
|
|
#endif
|
2017-09-05 17:29:57 +08:00
|
|
|
assert(mem[0]);
|
|
|
|
assert(mem[1]);
|
2017-07-20 16:26:35 +08:00
|
|
|
TaskHandle_t th[2];
|
|
|
|
err[0]=0; err[1]=0;
|
|
|
|
printf("Creating tasks\n");
|
|
|
|
xTaskCreatePinnedToCore(tstMem , "tskone" , 2048, mem[0], 3, &th[0], 0);
|
|
|
|
xTaskCreatePinnedToCore(tstMem , "tsktwo" , 2048, mem[1], 3, &th[1], 1);
|
|
|
|
|
|
|
|
for (int l=0; l<10; l++) {
|
|
|
|
for (int p=0; p<4096*1024; p+=65536) {
|
|
|
|
const void *out;
|
|
|
|
spi_flash_mmap_handle_t h;
|
|
|
|
spi_flash_mmap(p, 65536, SPI_FLASH_MMAP_DATA, &out, &h);
|
|
|
|
spi_flash_munmap(h);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
printf("Checked memory %d and %d times. Errors: %d and %d\n", res[0], res[1], err[0], err[1]);
|
|
|
|
|
|
|
|
vTaskDelete(th[0]);
|
|
|
|
vTaskDelete(th[1]);
|
2018-11-19 22:20:52 +05:30
|
|
|
#if USE_CAPS_ALLOC
|
2017-07-20 16:26:35 +08:00
|
|
|
free(mem[0]);
|
|
|
|
free(mem[1]);
|
|
|
|
#endif
|
|
|
|
TEST_ASSERT(err[0]==0);
|
|
|
|
TEST_ASSERT(err[1]==0);
|
|
|
|
}
|
|
|
|
|
|
|
|
#define CYCLES 1024
|
|
|
|
|
2018-11-19 22:20:52 +05:30
|
|
|
TEST_CASE("Spiram cache flush on write/read", "[spiram]")
|
2017-07-20 16:26:35 +08:00
|
|
|
{
|
|
|
|
void *mem[2];
|
|
|
|
res[0]=0; res[1]=0;
|
2018-11-19 22:20:52 +05:30
|
|
|
#if USE_CAPS_ALLOC
|
2017-09-06 16:13:40 +08:00
|
|
|
printf("Allocating SPI RAM chunk...\n");
|
2017-09-05 17:29:57 +08:00
|
|
|
mem[0]=heap_caps_malloc(TSTSZ, MALLOC_CAP_SPIRAM);
|
|
|
|
mem[1]=heap_caps_malloc(TSTSZ, MALLOC_CAP_SPIRAM);
|
2017-07-20 16:26:35 +08:00
|
|
|
#else
|
2019-04-08 18:02:05 +08:00
|
|
|
mem[0]=(void*)SOC_EXTRAM_DATA_LOW;
|
|
|
|
mem[1]=(void*)SOC_EXTRAM_DATA_LOW+TSTSZ;
|
2017-07-20 16:26:35 +08:00
|
|
|
#endif
|
2017-09-05 17:29:57 +08:00
|
|
|
assert(mem[0]);
|
|
|
|
assert(mem[1]);
|
2017-07-20 16:26:35 +08:00
|
|
|
TaskHandle_t th[2];
|
|
|
|
const esp_partition_t* part = get_test_data_partition();
|
|
|
|
assert(part!=NULL);
|
|
|
|
|
|
|
|
printf("Erasing sector...\n");
|
|
|
|
esp_partition_erase_range(part, 0, 64*1024);
|
|
|
|
printf("Erased.\n");
|
|
|
|
printf("Creating tasks\n");
|
|
|
|
xTaskCreatePinnedToCore(tstMem , "tskone" , 2048, mem[0], 3, &th[0], 0);
|
|
|
|
xTaskCreatePinnedToCore(tstMem , "tsktwo" , 2048, mem[1], 3, &th[1], 1);
|
|
|
|
char buf[512];
|
|
|
|
|
|
|
|
const void *out;
|
2022-11-07 19:31:29 +08:00
|
|
|
esp_partition_mmap_handle_t handle;
|
|
|
|
esp_partition_mmap(part, 0, 512, ESP_PARTITION_MMAP_DATA, &out, &handle);
|
2017-07-20 16:26:35 +08:00
|
|
|
for (int i=0; i<CYCLES; i++) {
|
|
|
|
esp_partition_write(part, 0, buf, 512);
|
|
|
|
esp_partition_read(part, 0, buf, 512);
|
|
|
|
vTaskDelay(1);
|
|
|
|
}
|
2022-11-07 19:31:29 +08:00
|
|
|
esp_partition_munmap(handle);
|
2017-07-20 16:26:35 +08:00
|
|
|
|
|
|
|
printf("Checked memory %d and %d times.\n", res[0], res[1]);
|
|
|
|
|
|
|
|
vTaskDelete(th[0]);
|
|
|
|
vTaskDelete(th[1]);
|
2018-11-19 22:20:52 +05:30
|
|
|
#if USE_CAPS_ALLOC
|
2017-07-20 16:26:35 +08:00
|
|
|
free(mem[0]);
|
|
|
|
free(mem[1]);
|
|
|
|
#endif
|
|
|
|
}
|
2018-11-19 22:20:52 +05:30
|
|
|
#endif // !CONFIG_FREERTOS_UNICORE
|
2017-07-20 16:26:35 +08:00
|
|
|
|
2018-11-19 22:20:52 +05:30
|
|
|
IRAM_ATTR TEST_CASE("Spiram memcmp weirdness at 80MHz", "[spiram]") {
|
2017-09-05 17:29:57 +08:00
|
|
|
char *mem1=malloc(0x10000);
|
2018-11-19 22:20:52 +05:30
|
|
|
#if USE_CAPS_ALLOC
|
2017-09-05 17:29:57 +08:00
|
|
|
char *mem2=heap_caps_malloc(0x10000, MALLOC_CAP_SPIRAM);
|
|
|
|
#else
|
2019-04-08 18:02:05 +08:00
|
|
|
char *mem2=(void*)SOC_EXTRAM_DATA_LOW;
|
2017-09-05 17:29:57 +08:00
|
|
|
#endif
|
|
|
|
|
|
|
|
#if !CONFIG_SPIRAM_SPEED_80M
|
|
|
|
printf("**** WARNING **** Spi memory isn't running at 80MHz, so this test is somewhat meaningless.\n");
|
|
|
|
#endif
|
|
|
|
|
|
|
|
printf("RAM: Got %p and %p\n", mem1, mem2);
|
|
|
|
assert(mem1);
|
|
|
|
assert(mem2);
|
|
|
|
for (int i=0; i<0x10000; i++) mem1[i]=i^0xAAAAAAAA;
|
2019-03-14 17:29:32 +08:00
|
|
|
|
2017-09-05 17:29:57 +08:00
|
|
|
for (int cycle=1; cycle<100; cycle++) {
|
|
|
|
memcpy(mem2, mem1, 0x10000);
|
|
|
|
if (memcmp(mem1, mem2, 0x10000)!=0) {
|
|
|
|
printf("Memcmp failed! Cycle %d\n", cycle);
|
|
|
|
for (int i=0; i<0x10000; i++) {
|
|
|
|
if (mem1[i]!=mem2[i]) {
|
|
|
|
printf("Found real difference at index %d: 0x%x vs 0x%x\n", i, mem1[i], mem2[i]);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2017-09-06 16:13:40 +08:00
|
|
|
|
2018-11-19 22:20:52 +05:30
|
|
|
free(mem1);
|
|
|
|
#if USE_CAPS_ALLOC
|
|
|
|
free(mem2);
|
|
|
|
#endif
|
|
|
|
}
|
2017-09-06 16:13:40 +08:00
|
|
|
|
2019-06-05 14:34:19 +10:00
|
|
|
#endif // CONFIG_SPIRAM
|