mirror of
https://github.com/google/pebble.git
synced 2025-07-21 23:04:51 -04:00
Import of the watch repository from Pebble
This commit is contained in:
commit
3b92768480
10334 changed files with 2564465 additions and 0 deletions
297
tests/fw/mfg/test_spalding_mfg_info.c
Normal file
297
tests/fw/mfg/test_spalding_mfg_info.c
Normal file
|
@ -0,0 +1,297 @@
|
|||
/*
|
||||
* 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);
|
||||
}
|
Loading…
Add table
Add a link
Reference in a new issue