зеркало из https://github.com/mozilla/gecko-dev.git
668 строки
18 KiB
C++
668 строки
18 KiB
C++
/* This Source Code Form is subject to the terms of the Mozilla Public
|
|
* License, v. 2.0. If a copy of the MPL was not distributed with this
|
|
* file, You can obtain one at http://mozilla.org/MPL/2.0/. */
|
|
|
|
#include "CacheIndex.h"
|
|
#include "CacheLog.h"
|
|
#include "CacheFileUtils.h"
|
|
#include "CacheObserver.h"
|
|
#include "LoadContextInfo.h"
|
|
#include "mozilla/Tokenizer.h"
|
|
#include "mozilla/Telemetry.h"
|
|
#include "nsCOMPtr.h"
|
|
#include "nsString.h"
|
|
#include <algorithm>
|
|
#include "mozilla/Unused.h"
|
|
|
|
namespace mozilla::net::CacheFileUtils {
|
|
|
|
// This designates the format for the "alt-data" metadata.
|
|
// When the format changes we need to update the version.
|
|
static uint32_t const kAltDataVersion = 1;
|
|
const char* kAltDataKey = "alt-data";
|
|
|
|
namespace {
|
|
|
|
/**
|
|
* A simple recursive descent parser for the mapping key.
|
|
*/
|
|
class KeyParser : protected Tokenizer {
|
|
public:
|
|
explicit KeyParser(nsACString const& aInput)
|
|
: Tokenizer(aInput),
|
|
isAnonymous(false)
|
|
// Initialize the cache key to a zero length by default
|
|
,
|
|
lastTag(0) {}
|
|
|
|
private:
|
|
// Results
|
|
OriginAttributes originAttribs;
|
|
bool isAnonymous;
|
|
nsCString idEnhance;
|
|
nsDependentCSubstring cacheKey;
|
|
|
|
// Keeps the last tag name, used for alphabetical sort checking
|
|
char lastTag;
|
|
|
|
// Classifier for the 'tag' character valid range.
|
|
// Explicitly using unsigned char as 127 is -1 when signed and it would only
|
|
// produce a warning.
|
|
static bool TagChar(const char aChar) {
|
|
unsigned char c = static_cast<unsigned char>(aChar);
|
|
return c >= ' ' && c <= '\x7f';
|
|
}
|
|
|
|
bool ParseTags() {
|
|
// Expects to be at the tag name or at the end
|
|
if (CheckEOF()) {
|
|
return true;
|
|
}
|
|
|
|
char tag;
|
|
if (!ReadChar(&TagChar, &tag)) {
|
|
return false;
|
|
}
|
|
|
|
// Check the alphabetical order, hard-fail on disobedience
|
|
if (!(lastTag < tag || tag == ':')) {
|
|
return false;
|
|
}
|
|
lastTag = tag;
|
|
|
|
switch (tag) {
|
|
case ':':
|
|
// last possible tag, when present there is the cacheKey following,
|
|
// not terminated with ',' and no need to unescape.
|
|
cacheKey.Rebind(mCursor, mEnd - mCursor);
|
|
return true;
|
|
case 'O': {
|
|
nsAutoCString originSuffix;
|
|
if (!ParseValue(&originSuffix) ||
|
|
!originAttribs.PopulateFromSuffix(originSuffix)) {
|
|
return false;
|
|
}
|
|
break;
|
|
}
|
|
case 'p':
|
|
originAttribs.SyncAttributesWithPrivateBrowsing(true);
|
|
break;
|
|
case 'b':
|
|
// Leaving to be able to read and understand oldformatted entries
|
|
originAttribs.mInIsolatedMozBrowser = true;
|
|
break;
|
|
case 'a':
|
|
isAnonymous = true;
|
|
break;
|
|
case 'i': {
|
|
// Leaving to be able to read and understand oldformatted entries
|
|
uint32_t deprecatedAppId = 0;
|
|
if (!ReadInteger(&deprecatedAppId)) {
|
|
return false; // not a valid 32-bit integer
|
|
}
|
|
break;
|
|
}
|
|
case '~':
|
|
if (!ParseValue(&idEnhance)) {
|
|
return false;
|
|
}
|
|
break;
|
|
default:
|
|
if (!ParseValue()) { // skip any tag values, optional
|
|
return false;
|
|
}
|
|
break;
|
|
}
|
|
|
|
// We expect a comma after every tag
|
|
if (!CheckChar(',')) {
|
|
return false;
|
|
}
|
|
|
|
// Recurse to the next tag
|
|
return ParseTags();
|
|
}
|
|
|
|
bool ParseValue(nsACString* result = nullptr) {
|
|
// If at the end, fail since we expect a comma ; value may be empty tho
|
|
if (CheckEOF()) {
|
|
return false;
|
|
}
|
|
|
|
Token t;
|
|
while (Next(t)) {
|
|
if (!Token::Char(',').Equals(t)) {
|
|
if (result) {
|
|
result->Append(t.Fragment());
|
|
}
|
|
continue;
|
|
}
|
|
|
|
if (CheckChar(',')) {
|
|
// Two commas in a row, escaping
|
|
if (result) {
|
|
result->Append(',');
|
|
}
|
|
continue;
|
|
}
|
|
|
|
// We must give the comma back since the upper calls expect it
|
|
Rollback();
|
|
return true;
|
|
}
|
|
|
|
return false;
|
|
}
|
|
|
|
public:
|
|
already_AddRefed<LoadContextInfo> Parse() {
|
|
RefPtr<LoadContextInfo> info;
|
|
if (ParseTags()) {
|
|
info = GetLoadContextInfo(isAnonymous, originAttribs);
|
|
}
|
|
|
|
return info.forget();
|
|
}
|
|
|
|
void URISpec(nsACString& result) { result.Assign(cacheKey); }
|
|
|
|
void IdEnhance(nsACString& result) { result.Assign(idEnhance); }
|
|
};
|
|
|
|
} // namespace
|
|
|
|
already_AddRefed<nsILoadContextInfo> ParseKey(const nsACString& aKey,
|
|
nsACString* aIdEnhance,
|
|
nsACString* aURISpec) {
|
|
KeyParser parser(aKey);
|
|
RefPtr<LoadContextInfo> info = parser.Parse();
|
|
|
|
if (info) {
|
|
if (aIdEnhance) parser.IdEnhance(*aIdEnhance);
|
|
if (aURISpec) parser.URISpec(*aURISpec);
|
|
}
|
|
|
|
return info.forget();
|
|
}
|
|
|
|
void AppendKeyPrefix(nsILoadContextInfo* aInfo, nsACString& _retval) {
|
|
/**
|
|
* This key is used to salt file hashes. When form of the key is changed
|
|
* cache entries will fail to find on disk.
|
|
*
|
|
* IMPORTANT NOTE:
|
|
* Keep the attributes list sorted according their ASCII code.
|
|
*/
|
|
|
|
if (!aInfo) {
|
|
return;
|
|
}
|
|
|
|
OriginAttributes const* oa = aInfo->OriginAttributesPtr();
|
|
nsAutoCString suffix;
|
|
oa->CreateSuffix(suffix);
|
|
if (!suffix.IsEmpty()) {
|
|
AppendTagWithValue(_retval, 'O', suffix);
|
|
}
|
|
|
|
if (aInfo->IsAnonymous()) {
|
|
_retval.AppendLiteral("a,");
|
|
}
|
|
|
|
if (aInfo->IsPrivate()) {
|
|
_retval.AppendLiteral("p,");
|
|
}
|
|
}
|
|
|
|
void AppendTagWithValue(nsACString& aTarget, char const aTag,
|
|
const nsACString& aValue) {
|
|
aTarget.Append(aTag);
|
|
|
|
// First check the value string to save some memory copying
|
|
// for cases we don't need to escape at all (most likely).
|
|
if (!aValue.IsEmpty()) {
|
|
if (!aValue.Contains(',')) {
|
|
// No need to escape
|
|
aTarget.Append(aValue);
|
|
} else {
|
|
nsAutoCString escapedValue(aValue);
|
|
escapedValue.ReplaceSubstring(","_ns, ",,"_ns);
|
|
aTarget.Append(escapedValue);
|
|
}
|
|
}
|
|
|
|
aTarget.Append(',');
|
|
}
|
|
|
|
nsresult KeyMatchesLoadContextInfo(const nsACString& aKey,
|
|
nsILoadContextInfo* aInfo, bool* _retval) {
|
|
nsCOMPtr<nsILoadContextInfo> info = ParseKey(aKey);
|
|
|
|
if (!info) {
|
|
return NS_ERROR_FAILURE;
|
|
}
|
|
|
|
*_retval = info->Equals(aInfo);
|
|
return NS_OK;
|
|
}
|
|
|
|
ValidityPair::ValidityPair(uint32_t aOffset, uint32_t aLen)
|
|
: mOffset(aOffset), mLen(aLen) {}
|
|
|
|
bool ValidityPair::CanBeMerged(const ValidityPair& aOther) const {
|
|
// The pairs can be merged into a single one if the start of one of the pairs
|
|
// is placed anywhere in the validity interval of other pair or exactly after
|
|
// its end.
|
|
return IsInOrFollows(aOther.mOffset) || aOther.IsInOrFollows(mOffset);
|
|
}
|
|
|
|
bool ValidityPair::IsInOrFollows(uint32_t aOffset) const {
|
|
return mOffset <= aOffset && mOffset + mLen >= aOffset;
|
|
}
|
|
|
|
bool ValidityPair::LessThan(const ValidityPair& aOther) const {
|
|
if (mOffset < aOther.mOffset) {
|
|
return true;
|
|
}
|
|
|
|
if (mOffset == aOther.mOffset && mLen < aOther.mLen) {
|
|
return true;
|
|
}
|
|
|
|
return false;
|
|
}
|
|
|
|
void ValidityPair::Merge(const ValidityPair& aOther) {
|
|
MOZ_ASSERT(CanBeMerged(aOther));
|
|
|
|
uint32_t offset = std::min(mOffset, aOther.mOffset);
|
|
uint32_t end = std::max(mOffset + mLen, aOther.mOffset + aOther.mLen);
|
|
|
|
mOffset = offset;
|
|
mLen = end - offset;
|
|
}
|
|
|
|
void ValidityMap::Log() const {
|
|
LOG(("ValidityMap::Log() - number of pairs: %zu", mMap.Length()));
|
|
for (uint32_t i = 0; i < mMap.Length(); i++) {
|
|
LOG((" (%u, %u)", mMap[i].Offset() + 0, mMap[i].Len() + 0));
|
|
}
|
|
}
|
|
|
|
uint32_t ValidityMap::Length() const { return mMap.Length(); }
|
|
|
|
void ValidityMap::AddPair(uint32_t aOffset, uint32_t aLen) {
|
|
ValidityPair pair(aOffset, aLen);
|
|
|
|
if (mMap.Length() == 0) {
|
|
mMap.AppendElement(pair);
|
|
return;
|
|
}
|
|
|
|
// Find out where to place this pair into the map, it can overlap only with
|
|
// one preceding pair and all subsequent pairs.
|
|
uint32_t pos = 0;
|
|
for (pos = mMap.Length(); pos > 0;) {
|
|
--pos;
|
|
|
|
if (mMap[pos].LessThan(pair)) {
|
|
// The new pair should be either inserted after pos or merged with it.
|
|
if (mMap[pos].CanBeMerged(pair)) {
|
|
// Merge with the preceding pair
|
|
mMap[pos].Merge(pair);
|
|
} else {
|
|
// They don't overlap, element must be placed after pos element
|
|
++pos;
|
|
if (pos == mMap.Length()) {
|
|
mMap.AppendElement(pair);
|
|
} else {
|
|
mMap.InsertElementAt(pos, pair);
|
|
}
|
|
}
|
|
|
|
break;
|
|
}
|
|
|
|
if (pos == 0) {
|
|
// The new pair should be placed in front of all existing pairs.
|
|
mMap.InsertElementAt(0, pair);
|
|
}
|
|
}
|
|
|
|
// pos now points to merged or inserted pair, check whether it overlaps with
|
|
// subsequent pairs.
|
|
while (pos + 1 < mMap.Length()) {
|
|
if (mMap[pos].CanBeMerged(mMap[pos + 1])) {
|
|
mMap[pos].Merge(mMap[pos + 1]);
|
|
mMap.RemoveElementAt(pos + 1);
|
|
} else {
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
|
|
void ValidityMap::Clear() { mMap.Clear(); }
|
|
|
|
size_t ValidityMap::SizeOfExcludingThis(
|
|
mozilla::MallocSizeOf mallocSizeOf) const {
|
|
return mMap.ShallowSizeOfExcludingThis(mallocSizeOf);
|
|
}
|
|
|
|
ValidityPair& ValidityMap::operator[](uint32_t aIdx) {
|
|
return mMap.ElementAt(aIdx);
|
|
}
|
|
|
|
StaticMutex DetailedCacheHitTelemetry::sLock;
|
|
uint32_t DetailedCacheHitTelemetry::sRecordCnt = 0;
|
|
DetailedCacheHitTelemetry::HitRate
|
|
DetailedCacheHitTelemetry::sHRStats[kNumOfRanges];
|
|
|
|
DetailedCacheHitTelemetry::HitRate::HitRate() { Reset(); }
|
|
|
|
void DetailedCacheHitTelemetry::HitRate::AddRecord(ERecType aType) {
|
|
if (aType == HIT) {
|
|
++mHitCnt;
|
|
} else {
|
|
++mMissCnt;
|
|
}
|
|
}
|
|
|
|
uint32_t DetailedCacheHitTelemetry::HitRate::GetHitRateBucket(
|
|
uint32_t aNumOfBuckets) const {
|
|
uint32_t bucketIdx = (aNumOfBuckets * mHitCnt) / (mHitCnt + mMissCnt);
|
|
if (bucketIdx ==
|
|
aNumOfBuckets) { // make sure 100% falls into the last bucket
|
|
--bucketIdx;
|
|
}
|
|
|
|
return bucketIdx;
|
|
}
|
|
|
|
uint32_t DetailedCacheHitTelemetry::HitRate::Count() {
|
|
return mHitCnt + mMissCnt;
|
|
}
|
|
|
|
void DetailedCacheHitTelemetry::HitRate::Reset() {
|
|
mHitCnt = 0;
|
|
mMissCnt = 0;
|
|
}
|
|
|
|
// static
|
|
void DetailedCacheHitTelemetry::AddRecord(ERecType aType,
|
|
TimeStamp aLoadStart) {
|
|
bool isUpToDate = false;
|
|
CacheIndex::IsUpToDate(&isUpToDate);
|
|
if (!isUpToDate) {
|
|
// Ignore the record when the entry file count might be incorrect
|
|
return;
|
|
}
|
|
|
|
uint32_t entryCount;
|
|
nsresult rv = CacheIndex::GetEntryFileCount(&entryCount);
|
|
if (NS_FAILED(rv)) {
|
|
return;
|
|
}
|
|
|
|
uint32_t rangeIdx = entryCount / kRangeSize;
|
|
if (rangeIdx >= kNumOfRanges) { // The last range has no upper limit.
|
|
rangeIdx = kNumOfRanges - 1;
|
|
}
|
|
|
|
uint32_t hitMissValue = 2 * rangeIdx; // 2 values per range
|
|
if (aType == MISS) { // The order is HIT, MISS
|
|
++hitMissValue;
|
|
}
|
|
|
|
StaticMutexAutoLock lock(sLock);
|
|
|
|
if (aType == MISS) {
|
|
mozilla::Telemetry::AccumulateTimeDelta(
|
|
mozilla::Telemetry::NETWORK_CACHE_V2_MISS_TIME_MS, aLoadStart);
|
|
} else {
|
|
mozilla::Telemetry::AccumulateTimeDelta(
|
|
mozilla::Telemetry::NETWORK_CACHE_V2_HIT_TIME_MS, aLoadStart);
|
|
}
|
|
|
|
Telemetry::Accumulate(Telemetry::NETWORK_CACHE_HIT_MISS_STAT_PER_CACHE_SIZE,
|
|
hitMissValue);
|
|
|
|
sHRStats[rangeIdx].AddRecord(aType);
|
|
++sRecordCnt;
|
|
|
|
if (sRecordCnt < kTotalSamplesReportLimit) {
|
|
return;
|
|
}
|
|
|
|
sRecordCnt = 0;
|
|
|
|
for (uint32_t i = 0; i < kNumOfRanges; ++i) {
|
|
if (sHRStats[i].Count() >= kHitRateSamplesReportLimit) {
|
|
// The telemetry enums are grouped by buckets as follows:
|
|
// Telemetry value : 0,1,2,3, ... ,19,20,21,22, ... ,398,399
|
|
// Hit rate bucket : 0,0,0,0, ... , 0, 1, 1, 1, ... , 19, 19
|
|
// Cache size range: 0,1,2,3, ... ,19, 0, 1, 2, ... , 18, 19
|
|
uint32_t bucketOffset =
|
|
sHRStats[i].GetHitRateBucket(kHitRateBuckets) * kNumOfRanges;
|
|
|
|
Telemetry::Accumulate(Telemetry::NETWORK_CACHE_HIT_RATE_PER_CACHE_SIZE,
|
|
bucketOffset + i);
|
|
sHRStats[i].Reset();
|
|
}
|
|
}
|
|
}
|
|
|
|
StaticMutex CachePerfStats::sLock;
|
|
CachePerfStats::PerfData CachePerfStats::sData[CachePerfStats::LAST];
|
|
uint32_t CachePerfStats::sCacheSlowCnt = 0;
|
|
uint32_t CachePerfStats::sCacheNotSlowCnt = 0;
|
|
|
|
CachePerfStats::MMA::MMA(uint32_t aTotalWeight, bool aFilter)
|
|
: mSum(0), mSumSq(0), mCnt(0), mWeight(aTotalWeight), mFilter(aFilter) {}
|
|
|
|
void CachePerfStats::MMA::AddValue(uint32_t aValue) {
|
|
if (mFilter) {
|
|
// Filter high spikes
|
|
uint32_t avg = GetAverage();
|
|
uint32_t stddev = GetStdDev();
|
|
uint32_t maxdiff = avg + (3 * stddev);
|
|
if (avg && aValue > avg + maxdiff) {
|
|
return;
|
|
}
|
|
}
|
|
|
|
if (mCnt < mWeight) {
|
|
// Compute arithmetic average until we have at least mWeight values
|
|
CheckedInt<uint64_t> newSumSq = CheckedInt<uint64_t>(aValue) * aValue;
|
|
newSumSq += mSumSq;
|
|
if (!newSumSq.isValid()) {
|
|
return; // ignore this value
|
|
}
|
|
mSumSq = newSumSq.value();
|
|
mSum += aValue;
|
|
++mCnt;
|
|
} else {
|
|
CheckedInt<uint64_t> newSumSq = mSumSq - mSumSq / mCnt;
|
|
newSumSq += static_cast<uint64_t>(aValue) * aValue;
|
|
if (!newSumSq.isValid()) {
|
|
return; // ignore this value
|
|
}
|
|
mSumSq = newSumSq.value();
|
|
|
|
// Compute modified moving average for more values:
|
|
// newAvg = ((weight - 1) * oldAvg + newValue) / weight
|
|
mSum -= GetAverage();
|
|
mSum += aValue;
|
|
}
|
|
}
|
|
|
|
uint32_t CachePerfStats::MMA::GetAverage() {
|
|
if (mCnt == 0) {
|
|
return 0;
|
|
}
|
|
|
|
return mSum / mCnt;
|
|
}
|
|
|
|
uint32_t CachePerfStats::MMA::GetStdDev() {
|
|
if (mCnt == 0) {
|
|
return 0;
|
|
}
|
|
|
|
uint32_t avg = GetAverage();
|
|
uint64_t avgSq = static_cast<uint64_t>(avg) * avg;
|
|
uint64_t variance = mSumSq / mCnt;
|
|
if (variance < avgSq) {
|
|
// Due to rounding error when using integer data type, it can happen that
|
|
// average of squares of the values is smaller than square of the average
|
|
// of the values. In this case fix mSumSq.
|
|
variance = avgSq;
|
|
mSumSq = variance * mCnt;
|
|
}
|
|
|
|
variance -= avgSq;
|
|
return sqrt(static_cast<double>(variance));
|
|
}
|
|
|
|
CachePerfStats::PerfData::PerfData()
|
|
: mFilteredAvg(50, true), mShortAvg(3, false) {}
|
|
|
|
void CachePerfStats::PerfData::AddValue(uint32_t aValue, bool aShortOnly) {
|
|
if (!aShortOnly) {
|
|
mFilteredAvg.AddValue(aValue);
|
|
}
|
|
mShortAvg.AddValue(aValue);
|
|
}
|
|
|
|
uint32_t CachePerfStats::PerfData::GetAverage(bool aFiltered) {
|
|
return aFiltered ? mFilteredAvg.GetAverage() : mShortAvg.GetAverage();
|
|
}
|
|
|
|
uint32_t CachePerfStats::PerfData::GetStdDev(bool aFiltered) {
|
|
return aFiltered ? mFilteredAvg.GetStdDev() : mShortAvg.GetStdDev();
|
|
}
|
|
|
|
// static
|
|
void CachePerfStats::AddValue(EDataType aType, uint32_t aValue,
|
|
bool aShortOnly) {
|
|
StaticMutexAutoLock lock(sLock);
|
|
sData[aType].AddValue(aValue, aShortOnly);
|
|
}
|
|
|
|
// static
|
|
uint32_t CachePerfStats::GetAverage(EDataType aType, bool aFiltered) {
|
|
StaticMutexAutoLock lock(sLock);
|
|
return sData[aType].GetAverage(aFiltered);
|
|
}
|
|
|
|
// static
|
|
uint32_t CachePerfStats::GetStdDev(EDataType aType, bool aFiltered) {
|
|
StaticMutexAutoLock lock(sLock);
|
|
return sData[aType].GetStdDev(aFiltered);
|
|
}
|
|
|
|
// static
|
|
bool CachePerfStats::IsCacheSlow() {
|
|
StaticMutexAutoLock lock(sLock);
|
|
|
|
// Compare mShortAvg with mFilteredAvg to find out whether cache is getting
|
|
// slower. Use only data about single IO operations because ENTRY_OPEN can be
|
|
// affected by more factors than a slow disk.
|
|
for (uint32_t i = 0; i < ENTRY_OPEN; ++i) {
|
|
if (i == IO_WRITE) {
|
|
// Skip this data type. IsCacheSlow is used for determining cache slowness
|
|
// when opening entries. Writes have low priority and it's normal that
|
|
// they are delayed a lot, but this doesn't necessarily affect opening
|
|
// cache entries.
|
|
continue;
|
|
}
|
|
|
|
uint32_t avgLong = sData[i].GetAverage(true);
|
|
if (avgLong == 0) {
|
|
// We have no perf data yet, skip this data type.
|
|
continue;
|
|
}
|
|
uint32_t avgShort = sData[i].GetAverage(false);
|
|
uint32_t stddevLong = sData[i].GetStdDev(true);
|
|
uint32_t maxdiff = avgLong + (3 * stddevLong);
|
|
|
|
if (avgShort > avgLong + maxdiff) {
|
|
LOG(
|
|
("CachePerfStats::IsCacheSlow() - result is slow based on perf "
|
|
"type %u [avgShort=%u, avgLong=%u, stddevLong=%u]",
|
|
i, avgShort, avgLong, stddevLong));
|
|
++sCacheSlowCnt;
|
|
return true;
|
|
}
|
|
}
|
|
|
|
++sCacheNotSlowCnt;
|
|
return false;
|
|
}
|
|
|
|
// static
|
|
void CachePerfStats::GetSlowStats(uint32_t* aSlow, uint32_t* aNotSlow) {
|
|
StaticMutexAutoLock lock(sLock);
|
|
*aSlow = sCacheSlowCnt;
|
|
*aNotSlow = sCacheNotSlowCnt;
|
|
}
|
|
|
|
void FreeBuffer(void* aBuf) {
|
|
#ifndef NS_FREE_PERMANENT_DATA
|
|
if (CacheObserver::ShuttingDown()) {
|
|
return;
|
|
}
|
|
#endif
|
|
|
|
free(aBuf);
|
|
}
|
|
|
|
nsresult ParseAlternativeDataInfo(const char* aInfo, int64_t* _offset,
|
|
nsACString* _type) {
|
|
// The format is: "1;12345,javascript/binary"
|
|
// <version>;<offset>,<type>
|
|
mozilla::Tokenizer p(aInfo, nullptr, "/");
|
|
uint32_t altDataVersion = 0;
|
|
int64_t altDataOffset = -1;
|
|
|
|
// The metadata format has a wrong version number.
|
|
if (!p.ReadInteger(&altDataVersion) || altDataVersion != kAltDataVersion) {
|
|
LOG(
|
|
("ParseAlternativeDataInfo() - altDataVersion=%u, "
|
|
"expectedVersion=%u",
|
|
altDataVersion, kAltDataVersion));
|
|
return NS_ERROR_NOT_AVAILABLE;
|
|
}
|
|
|
|
if (!p.CheckChar(';') || !p.ReadInteger(&altDataOffset) ||
|
|
!p.CheckChar(',')) {
|
|
return NS_ERROR_NOT_AVAILABLE;
|
|
}
|
|
|
|
// The requested alt-data representation is not available
|
|
if (altDataOffset < 0) {
|
|
return NS_ERROR_NOT_AVAILABLE;
|
|
}
|
|
|
|
if (_offset) {
|
|
*_offset = altDataOffset;
|
|
}
|
|
|
|
if (_type) {
|
|
mozilla::Unused << p.ReadUntil(Tokenizer::Token::EndOfFile(), *_type);
|
|
}
|
|
|
|
return NS_OK;
|
|
}
|
|
|
|
void BuildAlternativeDataInfo(const char* aInfo, int64_t aOffset,
|
|
nsACString& _retval) {
|
|
_retval.Truncate();
|
|
_retval.AppendInt(kAltDataVersion);
|
|
_retval.Append(';');
|
|
_retval.AppendInt(aOffset);
|
|
_retval.Append(',');
|
|
_retval.Append(aInfo);
|
|
}
|
|
|
|
} // namespace mozilla::net::CacheFileUtils
|