Sources/SPTPersistentCacheHeader.m (74 lines of code) (raw):
/*
Copyright (c) 2015-2021 Spotify AB.
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.
*/
#import <SPTPersistentCache/SPTPersistentCacheHeader.h>
#import "NSError+SPTPersistentCacheDomainErrors.h"
#include "crc32iso3309.h"
const SPTPersistentCacheMagicType SPTPersistentCacheMagicValue = 0x46545053; // SPTF
const size_t SPTPersistentCacheRecordHeaderSize = sizeof(SPTPersistentCacheRecordHeader);
_Static_assert(sizeof(SPTPersistentCacheRecordHeader) == 64,
"Struct SPTPersistentCacheRecordHeader has to be packed without padding");
_Static_assert(sizeof(SPTPersistentCacheRecordHeader) % 4 == 0,
"Struct size has to be multiple of 4");
NS_INLINE BOOL SPTPersistentCachePointerMagicAlignCheck(const void *ptr)
{
const unsigned align = _Alignof(SPTPersistentCacheMagicType);
uint64_t v = (uint64_t)(ptr);
return (v % align == 0);
}
SPTPersistentCacheRecordHeader SPTPersistentCacheRecordHeaderMake(uint64_t ttl,
uint64_t payloadSize,
uint64_t updateTime,
BOOL isLocked)
{
SPTPersistentCacheRecordHeader dummy;
memset(&dummy, 0, SPTPersistentCacheRecordHeaderSize);
SPTPersistentCacheRecordHeader *header = &dummy;
header->magic = SPTPersistentCacheMagicValue;
header->headerSize = (uint32_t)SPTPersistentCacheRecordHeaderSize;
header->refCount = (isLocked ? 1 : 0);
header->ttl = ttl;
header->payloadSizeBytes = payloadSize;
header->updateTimeSec = updateTime;
header->crc = SPTPersistentCacheCalculateHeaderCRC(header);
return dummy;
}
SPTPersistentCacheRecordHeader *SPTPersistentCacheGetHeaderFromData(void *data, size_t size)
{
if (size < SPTPersistentCacheRecordHeaderSize) {
return NULL;
}
return (SPTPersistentCacheRecordHeader *)data;
}
int /*SPTPersistentCacheLoadingError*/ SPTPersistentCacheValidateHeader(const SPTPersistentCacheRecordHeader *header)
{
if (header == NULL) {
return SPTPersistentCacheLoadingErrorInternalInconsistency;
}
// Check that header could be read according to alignment
if (!SPTPersistentCachePointerMagicAlignCheck(header)) {
return SPTPersistentCacheLoadingErrorHeaderAlignmentMismatch;
}
// 1. Check magic
if (header->magic != SPTPersistentCacheMagicValue) {
return SPTPersistentCacheLoadingErrorMagicMismatch;
}
// 2. Check CRC
uint32_t crc = SPTPersistentCacheCalculateHeaderCRC(header);
if (crc != header->crc) {
return SPTPersistentCacheLoadingErrorInvalidHeaderCRC;
}
// 3. Check header size
if (header->headerSize != SPTPersistentCacheRecordHeaderSize) {
return SPTPersistentCacheLoadingErrorWrongHeaderSize;
}
return -1;
}
NSError * SPTPersistentCacheCheckValidHeader(SPTPersistentCacheRecordHeader *header)
{
int code = SPTPersistentCacheValidateHeader(header);
if (code == -1) { // No error
return nil;
}
return [NSError spt_persistentDataCacheErrorWithCode:code];
}
uint32_t SPTPersistentCacheCalculateHeaderCRC(const SPTPersistentCacheRecordHeader *header)
{
if (header == NULL) {
return 0;
}
return spt_crc32((const uint8_t *)header, SPTPersistentCacheRecordHeaderSize - sizeof(header->crc));
}