mirror of
https://github.com/google/pebble.git
synced 2025-03-21 19:31:20 +00:00
298 lines
9.3 KiB
C
298 lines
9.3 KiB
C
|
/*
|
||
|
* Copyright 2024 Google LLC
|
||
|
*
|
||
|
* Licensed 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.
|
||
|
*/
|
||
|
|
||
|
#include "clar.h"
|
||
|
|
||
|
#include "applib/graphics/gtypes.h"
|
||
|
#include "mfg/mfg_info.h"
|
||
|
#include "mfg/snowy/mfg_private.h"
|
||
|
#include "flash_region/flash_region_s29vs.h"
|
||
|
|
||
|
#include "fake_spi_flash.c"
|
||
|
#include "stubs_logging.h"
|
||
|
#include "stubs_pbl_malloc.h"
|
||
|
|
||
|
// This will come from overrides and will include a smaller boot fpga image
|
||
|
#include "mfg/spalding/spalding_boot.fpga.auto.h"
|
||
|
|
||
|
// Stubs for fake_crc.c
|
||
|
#include "services/normal/filesystem/pfs.h"
|
||
|
int pfs_read(int fd, void *buf, size_t size) { return 0; }
|
||
|
int pfs_seek(int fd, int offset, FSeekType seek_type) { return 0; }
|
||
|
|
||
|
// Test Code!
|
||
|
|
||
|
void test_spalding_mfg_info__initialize(void) {
|
||
|
fake_spi_flash_init(FLASH_REGION_MFG_INFO_BEGIN,
|
||
|
FLASH_REGION_MFG_INFO_END - FLASH_REGION_MFG_INFO_BEGIN);
|
||
|
}
|
||
|
|
||
|
void test_spalding_mfg_info__cleanup(void) {
|
||
|
fake_spi_flash_cleanup();
|
||
|
}
|
||
|
|
||
|
void test_spalding_mfg_info__color(void) {
|
||
|
cl_assert_equal_i(mfg_info_get_watch_color(), 0);
|
||
|
|
||
|
mfg_info_set_watch_color(WATCH_INFO_COLOR_RED);
|
||
|
cl_assert_equal_i(mfg_info_get_watch_color(), WATCH_INFO_COLOR_RED);
|
||
|
|
||
|
mfg_info_set_watch_color(WATCH_INFO_COLOR_GREEN);
|
||
|
cl_assert_equal_i(mfg_info_get_watch_color(), WATCH_INFO_COLOR_GREEN);
|
||
|
}
|
||
|
|
||
|
void test_spalding_mfg_info__rtc_freq(void) {
|
||
|
cl_assert_equal_i(mfg_info_get_rtc_freq(), 0);
|
||
|
|
||
|
mfg_info_set_rtc_freq(0xfefefefe);
|
||
|
cl_assert_equal_i(mfg_info_get_rtc_freq(), 0xfefefefe);
|
||
|
|
||
|
mfg_info_set_rtc_freq(1337);
|
||
|
cl_assert_equal_i(mfg_info_get_rtc_freq(), 1337);
|
||
|
}
|
||
|
|
||
|
void test_spalding_mfg_info__model(void) {
|
||
|
// Intentionally make the buffer too long so we can check for truncation.
|
||
|
char buffer[MFG_INFO_MODEL_STRING_LENGTH + 1] = { 0 };
|
||
|
|
||
|
mfg_info_get_model(buffer);
|
||
|
cl_assert_equal_s(buffer, "");
|
||
|
|
||
|
mfg_info_set_model("test_model");
|
||
|
|
||
|
mfg_info_get_model(buffer);
|
||
|
cl_assert_equal_s(buffer, "test_model");
|
||
|
|
||
|
{
|
||
|
char long_string[] = "01234567890123456789";
|
||
|
mfg_info_set_model(long_string);
|
||
|
|
||
|
// We only expect to see the first 15 (MFG_INFO_MODEL_STRING_LENGTH - 1) characters
|
||
|
mfg_info_get_model(buffer);
|
||
|
cl_assert_equal_s(buffer, "012345678901234");
|
||
|
}
|
||
|
}
|
||
|
|
||
|
void test_spalding_mfg_info__1_to_2_conversion(void) {
|
||
|
// Force in an old data version.
|
||
|
typedef struct {
|
||
|
uint32_t data_version;
|
||
|
|
||
|
uint32_t color;
|
||
|
uint32_t rtc_freq;
|
||
|
} MfgDataV1;
|
||
|
|
||
|
MfgDataV1 old_data = {
|
||
|
.data_version = 1,
|
||
|
.color = 3,
|
||
|
.rtc_freq = 4
|
||
|
};
|
||
|
|
||
|
flash_write_bytes((const uint8_t*) &old_data, FLASH_REGION_MFG_INFO_BEGIN, sizeof(old_data));
|
||
|
|
||
|
// Now use the info functions to read the data and make sure it's sane. A conversion will have
|
||
|
// happened behind the scenes to the latest version.
|
||
|
cl_assert_equal_i(mfg_info_get_watch_color(), 3);
|
||
|
cl_assert_equal_i(mfg_info_get_rtc_freq(), 4);
|
||
|
|
||
|
char buffer[MFG_INFO_MODEL_STRING_LENGTH] = { 0 };
|
||
|
mfg_info_get_model(buffer);
|
||
|
cl_assert_equal_s(buffer, "");
|
||
|
|
||
|
// Set color and make sure others don't change.
|
||
|
mfg_info_set_watch_color(5);
|
||
|
|
||
|
cl_assert_equal_i(mfg_info_get_watch_color(), 5);
|
||
|
cl_assert_equal_i(mfg_info_get_rtc_freq(), 4);
|
||
|
|
||
|
mfg_info_get_model(buffer);
|
||
|
cl_assert_equal_s(buffer, "");
|
||
|
|
||
|
// Make sure we have space for the model.
|
||
|
mfg_info_set_model("test_model");
|
||
|
|
||
|
cl_assert_equal_i(mfg_info_get_watch_color(), 5);
|
||
|
cl_assert_equal_i(mfg_info_get_rtc_freq(), 4);
|
||
|
|
||
|
mfg_info_get_model(buffer);
|
||
|
cl_assert_equal_s(buffer, "test_model");
|
||
|
}
|
||
|
|
||
|
|
||
|
void test_spalding_mfg_info__1_to_3_conversion(void) {
|
||
|
// Force in an old data version.
|
||
|
typedef struct {
|
||
|
uint32_t data_version;
|
||
|
|
||
|
uint32_t color;
|
||
|
uint32_t rtc_freq;
|
||
|
} MfgDataV1;
|
||
|
|
||
|
MfgDataV1 old_data = {
|
||
|
.data_version = 1,
|
||
|
.color = 3,
|
||
|
.rtc_freq = 4
|
||
|
};
|
||
|
|
||
|
flash_write_bytes((const uint8_t*) &old_data, FLASH_REGION_MFG_INFO_BEGIN, sizeof(old_data));
|
||
|
|
||
|
// Now use the info functions to read the data and make sure it's sane. A conversion will have
|
||
|
// happened behind the scenes to the latest version.
|
||
|
cl_assert_equal_i(mfg_info_get_watch_color(), 3);
|
||
|
cl_assert_equal_i(mfg_info_get_rtc_freq(), 4);
|
||
|
|
||
|
char buffer[MFG_INFO_MODEL_STRING_LENGTH] = { 0 };
|
||
|
mfg_info_get_model(buffer);
|
||
|
cl_assert_equal_s(buffer, "");
|
||
|
|
||
|
cl_assert_equal_i(mfg_info_get_disp_offsets().x, 0);
|
||
|
cl_assert_equal_i(mfg_info_get_disp_offsets().y, 0);
|
||
|
|
||
|
// Set x and y offsets and make sure others don't change.
|
||
|
mfg_info_set_disp_offsets((GPoint) {-2, 1});
|
||
|
|
||
|
cl_assert_equal_i(mfg_info_get_disp_offsets().x, -2);
|
||
|
cl_assert_equal_i(mfg_info_get_disp_offsets().y, 1);
|
||
|
|
||
|
cl_assert_equal_i(mfg_info_get_watch_color(), 3);
|
||
|
cl_assert_equal_i(mfg_info_get_rtc_freq(), 4);
|
||
|
|
||
|
mfg_info_get_model(buffer);
|
||
|
cl_assert_equal_s(buffer, "");
|
||
|
|
||
|
// Make sure we have space for the model.
|
||
|
mfg_info_set_model("test_model");
|
||
|
|
||
|
cl_assert_equal_i(mfg_info_get_disp_offsets().x, -2);
|
||
|
cl_assert_equal_i(mfg_info_get_disp_offsets().y, 1);
|
||
|
|
||
|
cl_assert_equal_i(mfg_info_get_watch_color(), 3);
|
||
|
cl_assert_equal_i(mfg_info_get_rtc_freq(), 4);
|
||
|
|
||
|
mfg_info_get_model(buffer);
|
||
|
cl_assert_equal_s(buffer, "test_model");
|
||
|
}
|
||
|
|
||
|
void test_spalding_mfg_info__2_to_3_conversion(void) {
|
||
|
// Force in an old data version.
|
||
|
typedef struct {
|
||
|
uint32_t data_version;
|
||
|
|
||
|
uint32_t color;
|
||
|
uint32_t rtc_freq;
|
||
|
char model[MFG_INFO_MODEL_STRING_LENGTH];
|
||
|
} MfgDataV2;
|
||
|
|
||
|
MfgDataV2 old_data = {
|
||
|
.data_version = 1,
|
||
|
.color = 3,
|
||
|
.rtc_freq = 4,
|
||
|
.model[0] = '\0'
|
||
|
};
|
||
|
|
||
|
flash_write_bytes((const uint8_t*) &old_data, FLASH_REGION_MFG_INFO_BEGIN, sizeof(old_data));
|
||
|
|
||
|
// Now use the info functions to read the data and make sure it's sane. A conversion will have
|
||
|
// happened behind the scenes to the latest version.
|
||
|
cl_assert_equal_i(mfg_info_get_watch_color(), 3);
|
||
|
cl_assert_equal_i(mfg_info_get_rtc_freq(), 4);
|
||
|
|
||
|
char buffer[MFG_INFO_MODEL_STRING_LENGTH] = { 0 };
|
||
|
mfg_info_get_model(buffer);
|
||
|
cl_assert_equal_s(buffer, "");
|
||
|
|
||
|
cl_assert_equal_i(mfg_info_get_disp_offsets().x, 0);
|
||
|
cl_assert_equal_i(mfg_info_get_disp_offsets().y, 0);
|
||
|
|
||
|
// Set x and y offsets and make sure others don't change.
|
||
|
mfg_info_set_disp_offsets((GPoint) {-2, 1});
|
||
|
|
||
|
cl_assert_equal_i(mfg_info_get_disp_offsets().x, -2);
|
||
|
cl_assert_equal_i(mfg_info_get_disp_offsets().y, 1);
|
||
|
|
||
|
cl_assert_equal_i(mfg_info_get_watch_color(), 3);
|
||
|
cl_assert_equal_i(mfg_info_get_rtc_freq(), 4);
|
||
|
|
||
|
mfg_info_get_model(buffer);
|
||
|
cl_assert_equal_s(buffer, "");
|
||
|
|
||
|
// Make sure we have space for the model.
|
||
|
mfg_info_set_model("test_model");
|
||
|
|
||
|
cl_assert_equal_i(mfg_info_get_disp_offsets().x, -2);
|
||
|
cl_assert_equal_i(mfg_info_get_disp_offsets().y, 1);
|
||
|
|
||
|
cl_assert_equal_i(mfg_info_get_watch_color(), 3);
|
||
|
cl_assert_equal_i(mfg_info_get_rtc_freq(), 4);
|
||
|
|
||
|
mfg_info_get_model(buffer);
|
||
|
cl_assert_equal_s(buffer, "test_model");
|
||
|
}
|
||
|
|
||
|
void test_spalding_mfg_info__boot_fpga_persistence(void) {
|
||
|
// Make sure no FPGA image is stored
|
||
|
const uintptr_t BOOT_FPGA_FLASH_ADDR = FLASH_REGION_MFG_INFO_BEGIN + 0x10000;
|
||
|
const uint32_t HEADER_SIZE = 4; // sizeof(BootFPGAHeader)
|
||
|
|
||
|
uint8_t fpga_buffer[HEADER_SIZE + sizeof(s_boot_fpga)];
|
||
|
flash_read_bytes(fpga_buffer, BOOT_FPGA_FLASH_ADDR, sizeof(fpga_buffer));
|
||
|
for (int i = 0; i < sizeof(s_boot_fpga); ++i) {
|
||
|
cl_assert(fpga_buffer[i] == 0xff);
|
||
|
}
|
||
|
|
||
|
// Write some data in
|
||
|
mfg_info_set_rtc_freq(1);
|
||
|
|
||
|
// The first time we write something into mfg_info we'll actually write the boot fpga as a side
|
||
|
// effect. Make sure it's there.
|
||
|
flash_read_bytes(fpga_buffer, BOOT_FPGA_FLASH_ADDR, sizeof(fpga_buffer));
|
||
|
cl_assert(memcmp(fpga_buffer + HEADER_SIZE, s_boot_fpga, sizeof(s_boot_fpga)) == 0);
|
||
|
|
||
|
mfg_info_set_disp_offsets((GPoint) { 2, 3 });
|
||
|
|
||
|
char model[MFG_INFO_MODEL_STRING_LENGTH] = "123456789012345";
|
||
|
mfg_info_set_model(model);
|
||
|
|
||
|
// Now let's write in an fpga image
|
||
|
mfg_info_update_constant_data();
|
||
|
|
||
|
// Let's make sure the mfg data is still persisted
|
||
|
cl_assert_equal_i(mfg_info_get_rtc_freq(), 1);
|
||
|
cl_assert_equal_i(mfg_info_get_disp_offsets().x, 2);
|
||
|
cl_assert_equal_i(mfg_info_get_disp_offsets().y, 3);
|
||
|
|
||
|
char result_model[MFG_INFO_MODEL_STRING_LENGTH];
|
||
|
mfg_info_get_model(result_model);
|
||
|
cl_assert_equal_s(model, result_model);
|
||
|
|
||
|
// Make sure the boot fpga is still correct
|
||
|
flash_read_bytes(fpga_buffer, BOOT_FPGA_FLASH_ADDR, sizeof(fpga_buffer));
|
||
|
cl_assert(memcmp(fpga_buffer + HEADER_SIZE, s_boot_fpga, sizeof(s_boot_fpga)) == 0);
|
||
|
|
||
|
// Now invalidate the section and write it back. Make sure it comes back
|
||
|
flash_write_bytes((const uint8_t*) "xxxx", BOOT_FPGA_FLASH_ADDR + HEADER_SIZE, 4);
|
||
|
|
||
|
// Make sure it's corrupted
|
||
|
flash_read_bytes(fpga_buffer, BOOT_FPGA_FLASH_ADDR, sizeof(fpga_buffer));
|
||
|
cl_assert(memcmp(fpga_buffer + HEADER_SIZE, s_boot_fpga, sizeof(s_boot_fpga)) != 0);
|
||
|
|
||
|
// Now update it and make sure we healed the corruption
|
||
|
mfg_info_update_constant_data();
|
||
|
|
||
|
flash_read_bytes(fpga_buffer, BOOT_FPGA_FLASH_ADDR, sizeof(fpga_buffer));
|
||
|
cl_assert(memcmp(fpga_buffer + HEADER_SIZE, s_boot_fpga, sizeof(s_boot_fpga)) == 0);
|
||
|
}
|