mirror of
https://github.com/roytam1/palemoon27.git
synced 2026-05-26 14:18:48 +00:00
b808ffac2d
- Bug 1196391, part 3 - Make argument count assertions fatal in js::ExpandErrorArgumentsVA(). r=Waldo (8233c0afac) - minor cleanup (1da0b2c3e9) - Bug 1191765: Make Debugger.Object.prototype.getScript properly recognize functions without scripts. r=fitzgen (3e2753577f) - Bug 1165807 - display WeakSet and WeakMap contents in console; r=bz,fitzgen (d8f70d8e6a) - Bug 1226024 - Expose the root of the dominator tree to JavaScript; r=bz,sfink x # Please enter the commit message for your changes. Lines starting (1949832288) - Bug 1220702 - Part 1: Replace callback() and newNode() with variadic templates. What could go wrong? r=Waldo. (80e1b40871) - Bug 1220702 - Part 2: Fix the .method property of certain FunctionDeclaration nodes. r=Waldo. (47f244a6e0) - Bug 1220702 - Part 3: Distinguish ES6 generators from legacy generators in Reflect.parse() output. r=Waldo. (362bb8ea4b) - bit of Bug 1180017 - Fix up the badly-horked backout and re-land. (f4f92ff88c) - Bug 1155303 - Add telemetry for async DeferredFinalize max pause. r=smaug (3ddc7b8856) - Bug 1174796 - Make sure ReleaseNow releases everything. r=smaug (5065aa1f52) - Bug 1191918 - Remove printf debugging r=me (48e95e425e) - update some tests (26ae4a8050) - Bug 1218643 - correct a DOM test. r=smaug " (75c6302bbb) - space (26dfeca131) - Bug 1166805 part 1 - refactor common tests for whether an animated list mirrors the base version of the list into methods. r=dholbert (e538b90e00) - Bug 1166805 part 2 - Call SetCapacity before calling DOMSVGXXXList::MaybeInsertNullInAnimValListAt, to prevent fallible InsertElementAt calls from failing r=dholbert (918397681f) - Bug 1092125 - Part 1 - Add non-scaling-stroke support to nsSVGPathGeometryElement::GetGeometryBounds (except line). r=jwatt (19ee6f9517) - Bug 1092125 - part 2 - add non-scaling-stroke support to SVGLineElement::GetGeometryBounds. r=jwatt (2c84b88ff1) - Bug 1140080 - ensure we only create stop frames for gradients. r=dholbert (636db91975) - Bug 1182496 - Don't create frames for SVG <text> descendants with failing conditional processing attributes. r=dholbert (4e976c1587) - Bug 1149542 - Part 3: Crashtest. r=dholbert (cde5ca0f57) - Bug 1209525 - Protect GetGeometryBounds from a singular non-scaling-stroke transform. r=longsonr (2cd1f2e0a4) - Bug 325427 - Add crashtest. (98ab5e6907) - Bug 803562 - force -moz-appearance: none on foreignObject elements. r=dbaron (0ba37f76ad) - Bug 950324 - Add crashtest. (6e2f7bc4c2) - Bug 1178159 - Ignore stroke-linecap:"square" on circle and ellipse. r=longsonr (42f4a9a71c) - Bug 1187770 - work around draw targets that don't display zero-length lines. r=longsonr (a612616ecb) - Bug 1222812 - add a null check in case there is no old style. r=dholbert (6a883edea2) - Bug 958160 - Compute bounds in transformed space instead of user space in GetCoveredRegion. r=longsonr (d020a10c56) - Bug 1224061: Followup to fix b2g bustage r=me CLOSED TREE (1dc2693955) - Bug 1173573 - Fix possible crash initializing sessionstorage. r=honzab (c6c77ccf7d) - Bug 536509 - Update localStorage to use common StorageAllowedForWindow logic, r=ehsan (825ee71ff3) - Bug 606655 - delete cookies UI option AskMeEveryTime and its related comments and tests. r=mak (828dfe54a5) - Bug 1194052 - Append to redirectchain before asyncopen() is called (r=sicking,mayhemer) (bb051ceb94) - Bug 1145503 - TP exceptions added while in Private Browsing mode persist beyond the Private Browsing session. r=ehsan Import url-classifier and private browsing modules. (4492b2de09) - Bug 1138979 - Pref to turn TP on when in Private Browsing mode. r=mmc , r=ehsan (5078eaa914) - Bug 1168635 - Extend nsITLSServerSocket to customize cipher suites. r=keeler (185a551640) - Bug 1165423 - WebRTC Fix DTLS handshake by expanding UDP buffer. r=rjesup (1f207e03ee) - Bug 1219939 - make nsTemporaryFileInputStream nsISeekableStream, r=jduell (f2a5ddfbf2) - Bug 1125816 - Parse FTP directory listings of Windows CE and WEC7 FTP Server r=jduell (4137b29d21) - Bug 1171016 - Initialize the linelen variable at its declaration in ParseFTPList.cpp. r=mcmanus (2a3960897d) - Bug 1197313 - remove PR_snprintf calls in netwerk/; r=froydnj (005da76d31) - Bug 1219910 - make gSocketThread a relaxed atomic variable; r=mcmanus (620d299605) - Bug 1222867 - part 1 - return already_AddRefed from WebSocketEventService::CreateFrameIfNeeded; r=mcmanus (0eee829a08) - Bug 1222867 - part 2 - be smarter about transferring ownership of WebSocketFrame; r=mcmanus (4a9fd71798) - Bug 1211001 - constant ASSERTION: nsITimer->SetDelay() called when the one-shot timer is not set up, r=mcmanus (451c903cbe) - Bug 1130822 - properly decode arbitrarily aligned data for non-tier1 platforms. r=mcmanus (bcb99913dc) - Bug 1204731 - telemetry for peer h2 goaway r=hurley (d6748682b4) - Bug 1205810 - telemetry for local h2 goaway code r=hurley (e142625588) - bug 1194818 - h2 header priority handling r=hurley (51766fff44) - bug 1194820 - h2 push promise padding handling r=hurley (560ee1f480) - bug 1208114 - fix h2 connect tunnels r=hurley (d17f920c31) - Bug 1213060 (part 1) - Properly handle discarding padding in Http2Session::OnWriteSegment. r=mcmanus (2cce4ac006) - Bug 1213060 (part 2) - Re-add state assertion in Http2Session::OnWriteSegment. r=mcmanus (6b1c030780) - fix build (39845819f6) - Bug 1214076 - allow TokenServerClient errors to be JSON.stringify'd. r=rnewman (ab6085fa97) - Bug 1220007 P1 Allow ConsoleReportCollectors to flush to another collector. r=bz (a97b2c5a57) - Bug 1220007 P2 Make InterceptedChannel's collect logs locally and only flush to nsIChannel on main thread r=bz (85b77c5a44) - Bug 867407 - Fix cloning of file URIs with search query strings (r=sworkman) (da6fd51c15) - Bug 1220728 Clear pending exceptions if string conversion fails in SWintercept error handling. r=bz (b188c34862) - Bug 1147913 - Change NS_SOCKETTRANSPORTSERVICE_CONTRACTID to NS_STREAMTRANSPORTSERVICE_CONTRACTID in RespondWithHandler::ResolvedCallback. r=ehsan (5b443e88ab) - fix namespace (562ed51caa) - Bug 1206060 - Show pinning status at about:cache. r=michal (75ed53663f) - Bug 1032254 - Generic way to pin reasource in the HTTP cache, r=michal (eeb860f8e3) - Bug 1211504. Remove unused member from RefLayer. (fab6bae915)
1067 lines
28 KiB
C++
1067 lines
28 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 "CacheLog.h"
|
|
#include "CacheFileMetadata.h"
|
|
|
|
#include "CacheFileIOManager.h"
|
|
#include "nsICacheEntry.h"
|
|
#include "CacheHashUtils.h"
|
|
#include "CacheFileChunk.h"
|
|
#include "CacheFileUtils.h"
|
|
#include "nsILoadContextInfo.h"
|
|
#include "nsICacheEntry.h" // for nsICacheEntryMetaDataVisitor
|
|
#include "../cache/nsCacheUtils.h"
|
|
#include "nsIFile.h"
|
|
#include "mozilla/DebugOnly.h"
|
|
#include "prnetdb.h"
|
|
|
|
|
|
namespace mozilla {
|
|
namespace net {
|
|
|
|
#define kMinMetadataRead 1024 // TODO find optimal value from telemetry
|
|
#define kAlignSize 4096
|
|
|
|
// Most of the cache entries fit into one chunk due to current chunk size. Make
|
|
// sure to tweak this value if kChunkSize is going to change.
|
|
#define kInitialHashArraySize 1
|
|
|
|
// Initial elements buffer size.
|
|
#define kInitialBufSize 64
|
|
|
|
// Max size of elements in bytes.
|
|
#define kMaxElementsSize 64*1024
|
|
|
|
#define NOW_SECONDS() (uint32_t(PR_Now() / PR_USEC_PER_SEC))
|
|
|
|
NS_IMPL_ISUPPORTS(CacheFileMetadata, CacheFileIOListener)
|
|
|
|
CacheFileMetadata::CacheFileMetadata(CacheFileHandle *aHandle, const nsACString &aKey)
|
|
: CacheMemoryConsumer(NORMAL)
|
|
, mHandle(aHandle)
|
|
, mHashArray(nullptr)
|
|
, mHashArraySize(0)
|
|
, mHashCount(0)
|
|
, mOffset(-1)
|
|
, mBuf(nullptr)
|
|
, mBufSize(0)
|
|
, mWriteBuf(nullptr)
|
|
, mElementsSize(0)
|
|
, mIsDirty(false)
|
|
, mAnonymous(false)
|
|
, mAllocExactSize(false)
|
|
{
|
|
LOG(("CacheFileMetadata::CacheFileMetadata() [this=%p, handle=%p, key=%s]",
|
|
this, aHandle, PromiseFlatCString(aKey).get()));
|
|
|
|
MOZ_COUNT_CTOR(CacheFileMetadata);
|
|
memset(&mMetaHdr, 0, sizeof(CacheFileMetadataHeader));
|
|
mMetaHdr.mVersion = kCacheEntryVersion;
|
|
mMetaHdr.mExpirationTime = nsICacheEntry::NO_EXPIRATION_TIME;
|
|
mKey = aKey;
|
|
|
|
DebugOnly<nsresult> rv;
|
|
rv = ParseKey(aKey);
|
|
MOZ_ASSERT(NS_SUCCEEDED(rv));
|
|
}
|
|
|
|
CacheFileMetadata::CacheFileMetadata(bool aMemoryOnly, bool aPinned, const nsACString &aKey)
|
|
: CacheMemoryConsumer(aMemoryOnly ? MEMORY_ONLY : NORMAL)
|
|
, mHandle(nullptr)
|
|
, mHashArray(nullptr)
|
|
, mHashArraySize(0)
|
|
, mHashCount(0)
|
|
, mOffset(0)
|
|
, mBuf(nullptr)
|
|
, mBufSize(0)
|
|
, mWriteBuf(nullptr)
|
|
, mElementsSize(0)
|
|
, mIsDirty(true)
|
|
, mAnonymous(false)
|
|
, mAllocExactSize(false)
|
|
{
|
|
LOG(("CacheFileMetadata::CacheFileMetadata() [this=%p, key=%s]",
|
|
this, PromiseFlatCString(aKey).get()));
|
|
|
|
MOZ_COUNT_CTOR(CacheFileMetadata);
|
|
memset(&mMetaHdr, 0, sizeof(CacheFileMetadataHeader));
|
|
mMetaHdr.mVersion = kCacheEntryVersion;
|
|
if (aPinned) {
|
|
AddFlags(kCacheEntryIsPinned);
|
|
}
|
|
mMetaHdr.mExpirationTime = nsICacheEntry::NO_EXPIRATION_TIME;
|
|
mKey = aKey;
|
|
mMetaHdr.mKeySize = mKey.Length();
|
|
|
|
DebugOnly<nsresult> rv;
|
|
rv = ParseKey(aKey);
|
|
MOZ_ASSERT(NS_SUCCEEDED(rv));
|
|
}
|
|
|
|
CacheFileMetadata::CacheFileMetadata()
|
|
: CacheMemoryConsumer(DONT_REPORT /* This is a helper class */)
|
|
, mHandle(nullptr)
|
|
, mHashArray(nullptr)
|
|
, mHashArraySize(0)
|
|
, mHashCount(0)
|
|
, mOffset(0)
|
|
, mBuf(nullptr)
|
|
, mBufSize(0)
|
|
, mWriteBuf(nullptr)
|
|
, mElementsSize(0)
|
|
, mIsDirty(false)
|
|
, mAnonymous(false)
|
|
, mAllocExactSize(false)
|
|
{
|
|
LOG(("CacheFileMetadata::CacheFileMetadata() [this=%p]", this));
|
|
|
|
MOZ_COUNT_CTOR(CacheFileMetadata);
|
|
memset(&mMetaHdr, 0, sizeof(CacheFileMetadataHeader));
|
|
}
|
|
|
|
CacheFileMetadata::~CacheFileMetadata()
|
|
{
|
|
LOG(("CacheFileMetadata::~CacheFileMetadata() [this=%p]", this));
|
|
|
|
MOZ_COUNT_DTOR(CacheFileMetadata);
|
|
MOZ_ASSERT(!mListener);
|
|
|
|
if (mHashArray) {
|
|
free(mHashArray);
|
|
mHashArray = nullptr;
|
|
mHashArraySize = 0;
|
|
}
|
|
|
|
if (mBuf) {
|
|
free(mBuf);
|
|
mBuf = nullptr;
|
|
mBufSize = 0;
|
|
}
|
|
}
|
|
|
|
void
|
|
CacheFileMetadata::SetHandle(CacheFileHandle *aHandle)
|
|
{
|
|
LOG(("CacheFileMetadata::SetHandle() [this=%p, handle=%p]", this, aHandle));
|
|
|
|
MOZ_ASSERT(!mHandle);
|
|
|
|
mHandle = aHandle;
|
|
}
|
|
|
|
nsresult
|
|
CacheFileMetadata::GetKey(nsACString &_retval)
|
|
{
|
|
_retval = mKey;
|
|
return NS_OK;
|
|
}
|
|
|
|
nsresult
|
|
CacheFileMetadata::ReadMetadata(CacheFileMetadataListener *aListener)
|
|
{
|
|
LOG(("CacheFileMetadata::ReadMetadata() [this=%p, listener=%p]", this, aListener));
|
|
|
|
MOZ_ASSERT(!mListener);
|
|
MOZ_ASSERT(!mHashArray);
|
|
MOZ_ASSERT(!mBuf);
|
|
MOZ_ASSERT(!mWriteBuf);
|
|
|
|
nsresult rv;
|
|
|
|
int64_t size = mHandle->FileSize();
|
|
MOZ_ASSERT(size != -1);
|
|
|
|
if (size == 0) {
|
|
// this is a new entry
|
|
LOG(("CacheFileMetadata::ReadMetadata() - Filesize == 0, creating empty "
|
|
"metadata. [this=%p]", this));
|
|
|
|
InitEmptyMetadata();
|
|
aListener->OnMetadataRead(NS_OK);
|
|
return NS_OK;
|
|
}
|
|
|
|
if (size < int64_t(sizeof(CacheFileMetadataHeader) + 2*sizeof(uint32_t))) {
|
|
// there must be at least checksum, header and offset
|
|
LOG(("CacheFileMetadata::ReadMetadata() - File is corrupted, creating "
|
|
"empty metadata. [this=%p, filesize=%lld]", this, size));
|
|
|
|
InitEmptyMetadata();
|
|
aListener->OnMetadataRead(NS_OK);
|
|
return NS_OK;
|
|
}
|
|
|
|
// Set offset so that we read at least kMinMetadataRead if the file is big
|
|
// enough.
|
|
int64_t offset;
|
|
if (size < kMinMetadataRead) {
|
|
offset = 0;
|
|
} else {
|
|
offset = size - kMinMetadataRead;
|
|
}
|
|
|
|
// round offset to kAlignSize blocks
|
|
offset = (offset / kAlignSize) * kAlignSize;
|
|
|
|
mBufSize = size - offset;
|
|
mBuf = static_cast<char *>(moz_xmalloc(mBufSize));
|
|
|
|
DoMemoryReport(MemoryUsage());
|
|
|
|
LOG(("CacheFileMetadata::ReadMetadata() - Reading metadata from disk, trying "
|
|
"offset=%lld, filesize=%lld [this=%p]", offset, size, this));
|
|
|
|
mListener = aListener;
|
|
rv = CacheFileIOManager::Read(mHandle, offset, mBuf, mBufSize, this);
|
|
if (NS_FAILED(rv)) {
|
|
LOG(("CacheFileMetadata::ReadMetadata() - CacheFileIOManager::Read() failed"
|
|
" synchronously, creating empty metadata. [this=%p, rv=0x%08x]",
|
|
this, rv));
|
|
|
|
mListener = nullptr;
|
|
InitEmptyMetadata();
|
|
aListener->OnMetadataRead(NS_OK);
|
|
return NS_OK;
|
|
}
|
|
|
|
return NS_OK;
|
|
}
|
|
|
|
uint32_t
|
|
CacheFileMetadata::CalcMetadataSize(uint32_t aElementsSize, uint32_t aHashCount)
|
|
{
|
|
return sizeof(uint32_t) + // hash of the metadata
|
|
aHashCount * sizeof(CacheHash::Hash16_t) + // array of chunk hashes
|
|
sizeof(CacheFileMetadataHeader) + // metadata header
|
|
mKey.Length() + 1 + // key with trailing null
|
|
aElementsSize + // elements
|
|
sizeof(uint32_t); // offset
|
|
}
|
|
|
|
nsresult
|
|
CacheFileMetadata::WriteMetadata(uint32_t aOffset,
|
|
CacheFileMetadataListener *aListener)
|
|
{
|
|
LOG(("CacheFileMetadata::WriteMetadata() [this=%p, offset=%d, listener=%p]",
|
|
this, aOffset, aListener));
|
|
|
|
MOZ_ASSERT(!mListener);
|
|
MOZ_ASSERT(!mWriteBuf);
|
|
|
|
nsresult rv;
|
|
|
|
mIsDirty = false;
|
|
|
|
mWriteBuf = static_cast<char *>(moz_xmalloc(CalcMetadataSize(mElementsSize,
|
|
mHashCount)));
|
|
if (!mWriteBuf) {
|
|
return NS_ERROR_OUT_OF_MEMORY;
|
|
}
|
|
|
|
char *p = mWriteBuf + sizeof(uint32_t);
|
|
memcpy(p, mHashArray, mHashCount * sizeof(CacheHash::Hash16_t));
|
|
p += mHashCount * sizeof(CacheHash::Hash16_t);
|
|
mMetaHdr.WriteToBuf(p);
|
|
p += sizeof(CacheFileMetadataHeader);
|
|
memcpy(p, mKey.get(), mKey.Length());
|
|
p += mKey.Length();
|
|
*p = 0;
|
|
p++;
|
|
memcpy(p, mBuf, mElementsSize);
|
|
p += mElementsSize;
|
|
|
|
CacheHash::Hash32_t hash;
|
|
hash = CacheHash::Hash(mWriteBuf + sizeof(uint32_t),
|
|
p - mWriteBuf - sizeof(uint32_t));
|
|
NetworkEndian::writeUint32(mWriteBuf, hash);
|
|
|
|
NetworkEndian::writeUint32(p, aOffset);
|
|
p += sizeof(uint32_t);
|
|
|
|
char * writeBuffer = mWriteBuf;
|
|
if (aListener) {
|
|
mListener = aListener;
|
|
} else {
|
|
// We are not going to pass |this| as a callback so the buffer will be
|
|
// released by CacheFileIOManager. Just null out mWriteBuf here.
|
|
mWriteBuf = nullptr;
|
|
}
|
|
|
|
rv = CacheFileIOManager::Write(mHandle, aOffset, writeBuffer, p - writeBuffer,
|
|
true, true, aListener ? this : nullptr);
|
|
if (NS_FAILED(rv)) {
|
|
LOG(("CacheFileMetadata::WriteMetadata() - CacheFileIOManager::Write() "
|
|
"failed synchronously. [this=%p, rv=0x%08x]", this, rv));
|
|
|
|
mListener = nullptr;
|
|
if (mWriteBuf) {
|
|
free(mWriteBuf);
|
|
mWriteBuf = nullptr;
|
|
}
|
|
NS_ENSURE_SUCCESS(rv, rv);
|
|
}
|
|
|
|
DoMemoryReport(MemoryUsage());
|
|
|
|
return NS_OK;
|
|
}
|
|
|
|
nsresult
|
|
CacheFileMetadata::SyncReadMetadata(nsIFile *aFile)
|
|
{
|
|
LOG(("CacheFileMetadata::SyncReadMetadata() [this=%p]", this));
|
|
|
|
MOZ_ASSERT(!mListener);
|
|
MOZ_ASSERT(!mHandle);
|
|
MOZ_ASSERT(!mHashArray);
|
|
MOZ_ASSERT(!mBuf);
|
|
MOZ_ASSERT(!mWriteBuf);
|
|
MOZ_ASSERT(mKey.IsEmpty());
|
|
|
|
nsresult rv;
|
|
|
|
int64_t fileSize;
|
|
rv = aFile->GetFileSize(&fileSize);
|
|
if (NS_FAILED(rv)) {
|
|
// Don't bloat the console
|
|
return rv;
|
|
}
|
|
|
|
PRFileDesc *fd;
|
|
rv = aFile->OpenNSPRFileDesc(PR_RDONLY, 0600, &fd);
|
|
NS_ENSURE_SUCCESS(rv, rv);
|
|
|
|
int64_t offset = PR_Seek64(fd, fileSize - sizeof(uint32_t), PR_SEEK_SET);
|
|
if (offset == -1) {
|
|
PR_Close(fd);
|
|
return NS_ERROR_FAILURE;
|
|
}
|
|
|
|
uint32_t metaOffset;
|
|
int32_t bytesRead = PR_Read(fd, &metaOffset, sizeof(uint32_t));
|
|
if (bytesRead != sizeof(uint32_t)) {
|
|
PR_Close(fd);
|
|
return NS_ERROR_FAILURE;
|
|
}
|
|
|
|
metaOffset = NetworkEndian::readUint32(&metaOffset);
|
|
if (metaOffset > fileSize) {
|
|
PR_Close(fd);
|
|
return NS_ERROR_FAILURE;
|
|
}
|
|
|
|
mBuf = static_cast<char *>(moz_xmalloc(fileSize - metaOffset));
|
|
if (!mBuf) {
|
|
return NS_ERROR_OUT_OF_MEMORY;
|
|
}
|
|
mBufSize = fileSize - metaOffset;
|
|
|
|
DoMemoryReport(MemoryUsage());
|
|
|
|
offset = PR_Seek64(fd, metaOffset, PR_SEEK_SET);
|
|
if (offset == -1) {
|
|
PR_Close(fd);
|
|
return NS_ERROR_FAILURE;
|
|
}
|
|
|
|
bytesRead = PR_Read(fd, mBuf, mBufSize);
|
|
PR_Close(fd);
|
|
if (bytesRead != static_cast<int32_t>(mBufSize)) {
|
|
return NS_ERROR_FAILURE;
|
|
}
|
|
|
|
rv = ParseMetadata(metaOffset, 0, false);
|
|
NS_ENSURE_SUCCESS(rv, rv);
|
|
|
|
return NS_OK;
|
|
}
|
|
|
|
const char *
|
|
CacheFileMetadata::GetElement(const char *aKey)
|
|
{
|
|
const char *data = mBuf;
|
|
const char *limit = mBuf + mElementsSize;
|
|
|
|
while (data < limit) {
|
|
// Point to the value part
|
|
const char *value = data + strlen(data) + 1;
|
|
MOZ_ASSERT(value < limit, "Metadata elements corrupted");
|
|
if (strcmp(data, aKey) == 0) {
|
|
LOG(("CacheFileMetadata::GetElement() - Key found [this=%p, key=%s]",
|
|
this, aKey));
|
|
return value;
|
|
}
|
|
|
|
// Skip value part
|
|
data = value + strlen(value) + 1;
|
|
}
|
|
MOZ_ASSERT(data == limit, "Metadata elements corrupted");
|
|
LOG(("CacheFileMetadata::GetElement() - Key not found [this=%p, key=%s]",
|
|
this, aKey));
|
|
return nullptr;
|
|
}
|
|
|
|
nsresult
|
|
CacheFileMetadata::SetElement(const char *aKey, const char *aValue)
|
|
{
|
|
LOG(("CacheFileMetadata::SetElement() [this=%p, key=%s, value=%p]",
|
|
this, aKey, aValue));
|
|
|
|
MarkDirty();
|
|
|
|
nsresult rv;
|
|
|
|
const uint32_t keySize = strlen(aKey) + 1;
|
|
char *pos = const_cast<char *>(GetElement(aKey));
|
|
|
|
if (!aValue) {
|
|
// No value means remove the key/value pair completely, if existing
|
|
if (pos) {
|
|
uint32_t oldValueSize = strlen(pos) + 1;
|
|
uint32_t offset = pos - mBuf;
|
|
uint32_t remainder = mElementsSize - (offset + oldValueSize);
|
|
|
|
memmove(pos - keySize, pos + oldValueSize, remainder);
|
|
mElementsSize -= keySize + oldValueSize;
|
|
}
|
|
return NS_OK;
|
|
}
|
|
|
|
const uint32_t valueSize = strlen(aValue) + 1;
|
|
uint32_t newSize = mElementsSize + valueSize;
|
|
if (pos) {
|
|
const uint32_t oldValueSize = strlen(pos) + 1;
|
|
const uint32_t offset = pos - mBuf;
|
|
const uint32_t remainder = mElementsSize - (offset + oldValueSize);
|
|
|
|
// Update the value in place
|
|
newSize -= oldValueSize;
|
|
rv = EnsureBuffer(newSize);
|
|
if (NS_FAILED(rv)) {
|
|
return rv;
|
|
}
|
|
|
|
// Move the remainder to the right place
|
|
pos = mBuf + offset;
|
|
memmove(pos + valueSize, pos + oldValueSize, remainder);
|
|
} else {
|
|
// allocate new meta data element
|
|
newSize += keySize;
|
|
rv = EnsureBuffer(newSize);
|
|
if (NS_FAILED(rv)) {
|
|
return rv;
|
|
}
|
|
|
|
// Add after last element
|
|
pos = mBuf + mElementsSize;
|
|
memcpy(pos, aKey, keySize);
|
|
pos += keySize;
|
|
}
|
|
|
|
// Update value
|
|
memcpy(pos, aValue, valueSize);
|
|
mElementsSize = newSize;
|
|
|
|
return NS_OK;
|
|
}
|
|
|
|
nsresult
|
|
CacheFileMetadata::Visit(nsICacheEntryMetaDataVisitor *aVisitor)
|
|
{
|
|
const char *data = mBuf;
|
|
const char *limit = mBuf + mElementsSize;
|
|
|
|
while (data < limit) {
|
|
// Point to the value part
|
|
const char *value = data + strlen(data) + 1;
|
|
MOZ_ASSERT(value < limit, "Metadata elements corrupted");
|
|
|
|
aVisitor->OnMetaDataElement(data, value);
|
|
|
|
// Skip value part
|
|
data = value + strlen(value) + 1;
|
|
}
|
|
|
|
MOZ_ASSERT(data == limit, "Metadata elements corrupted");
|
|
|
|
return NS_OK;
|
|
}
|
|
|
|
CacheHash::Hash16_t
|
|
CacheFileMetadata::GetHash(uint32_t aIndex)
|
|
{
|
|
MOZ_ASSERT(aIndex < mHashCount);
|
|
return NetworkEndian::readUint16(&mHashArray[aIndex]);
|
|
}
|
|
|
|
nsresult
|
|
CacheFileMetadata::SetHash(uint32_t aIndex, CacheHash::Hash16_t aHash)
|
|
{
|
|
LOG(("CacheFileMetadata::SetHash() [this=%p, idx=%d, hash=%x]",
|
|
this, aIndex, aHash));
|
|
|
|
MarkDirty();
|
|
|
|
MOZ_ASSERT(aIndex <= mHashCount);
|
|
|
|
if (aIndex > mHashCount) {
|
|
return NS_ERROR_INVALID_ARG;
|
|
} else if (aIndex == mHashCount) {
|
|
if ((aIndex + 1) * sizeof(CacheHash::Hash16_t) > mHashArraySize) {
|
|
// reallocate hash array buffer
|
|
if (mHashArraySize == 0) {
|
|
mHashArraySize = kInitialHashArraySize * sizeof(CacheHash::Hash16_t);
|
|
} else {
|
|
mHashArraySize *= 2;
|
|
}
|
|
mHashArray = static_cast<CacheHash::Hash16_t *>(
|
|
moz_xrealloc(mHashArray, mHashArraySize));
|
|
}
|
|
|
|
mHashCount++;
|
|
}
|
|
|
|
NetworkEndian::writeUint16(&mHashArray[aIndex], aHash);
|
|
|
|
DoMemoryReport(MemoryUsage());
|
|
|
|
return NS_OK;
|
|
}
|
|
|
|
nsresult
|
|
CacheFileMetadata::AddFlags(uint32_t aFlags)
|
|
{
|
|
MarkDirty(false);
|
|
mMetaHdr.mFlags |= aFlags;
|
|
return NS_OK;
|
|
}
|
|
|
|
nsresult
|
|
CacheFileMetadata::RemoveFlags(uint32_t aFlags)
|
|
{
|
|
MarkDirty(false);
|
|
mMetaHdr.mFlags &= ~aFlags;
|
|
return NS_OK;
|
|
}
|
|
|
|
nsresult
|
|
CacheFileMetadata::GetFlags(uint32_t *_retval)
|
|
{
|
|
*_retval = mMetaHdr.mFlags;
|
|
return NS_OK;
|
|
}
|
|
|
|
nsresult
|
|
CacheFileMetadata::SetExpirationTime(uint32_t aExpirationTime)
|
|
{
|
|
LOG(("CacheFileMetadata::SetExpirationTime() [this=%p, expirationTime=%d]",
|
|
this, aExpirationTime));
|
|
|
|
MarkDirty(false);
|
|
mMetaHdr.mExpirationTime = aExpirationTime;
|
|
return NS_OK;
|
|
}
|
|
|
|
nsresult
|
|
CacheFileMetadata::GetExpirationTime(uint32_t *_retval)
|
|
{
|
|
*_retval = mMetaHdr.mExpirationTime;
|
|
return NS_OK;
|
|
}
|
|
|
|
nsresult
|
|
CacheFileMetadata::SetFrecency(uint32_t aFrecency)
|
|
{
|
|
LOG(("CacheFileMetadata::SetFrecency() [this=%p, frecency=%f]",
|
|
this, (double)aFrecency));
|
|
|
|
MarkDirty(false);
|
|
mMetaHdr.mFrecency = aFrecency;
|
|
return NS_OK;
|
|
}
|
|
|
|
nsresult
|
|
CacheFileMetadata::GetFrecency(uint32_t *_retval)
|
|
{
|
|
*_retval = mMetaHdr.mFrecency;
|
|
return NS_OK;
|
|
}
|
|
|
|
nsresult
|
|
CacheFileMetadata::GetLastModified(uint32_t *_retval)
|
|
{
|
|
*_retval = mMetaHdr.mLastModified;
|
|
return NS_OK;
|
|
}
|
|
|
|
nsresult
|
|
CacheFileMetadata::GetLastFetched(uint32_t *_retval)
|
|
{
|
|
*_retval = mMetaHdr.mLastFetched;
|
|
return NS_OK;
|
|
}
|
|
|
|
nsresult
|
|
CacheFileMetadata::GetFetchCount(uint32_t *_retval)
|
|
{
|
|
*_retval = mMetaHdr.mFetchCount;
|
|
return NS_OK;
|
|
}
|
|
|
|
nsresult
|
|
CacheFileMetadata::OnFetched()
|
|
{
|
|
MarkDirty(false);
|
|
|
|
mMetaHdr.mLastFetched = NOW_SECONDS();
|
|
++mMetaHdr.mFetchCount;
|
|
return NS_OK;
|
|
}
|
|
|
|
void
|
|
CacheFileMetadata::MarkDirty(bool aUpdateLastModified)
|
|
{
|
|
mIsDirty = true;
|
|
if (aUpdateLastModified) {
|
|
mMetaHdr.mLastModified = NOW_SECONDS();
|
|
}
|
|
}
|
|
|
|
nsresult
|
|
CacheFileMetadata::OnFileOpened(CacheFileHandle *aHandle, nsresult aResult)
|
|
{
|
|
MOZ_CRASH("CacheFileMetadata::OnFileOpened should not be called!");
|
|
return NS_ERROR_UNEXPECTED;
|
|
}
|
|
|
|
nsresult
|
|
CacheFileMetadata::OnDataWritten(CacheFileHandle *aHandle, const char *aBuf,
|
|
nsresult aResult)
|
|
{
|
|
LOG(("CacheFileMetadata::OnDataWritten() [this=%p, handle=%p, result=0x%08x]",
|
|
this, aHandle, aResult));
|
|
|
|
MOZ_ASSERT(mListener);
|
|
MOZ_ASSERT(mWriteBuf);
|
|
|
|
free(mWriteBuf);
|
|
mWriteBuf = nullptr;
|
|
|
|
nsCOMPtr<CacheFileMetadataListener> listener;
|
|
|
|
mListener.swap(listener);
|
|
listener->OnMetadataWritten(aResult);
|
|
|
|
DoMemoryReport(MemoryUsage());
|
|
|
|
return NS_OK;
|
|
}
|
|
|
|
nsresult
|
|
CacheFileMetadata::OnDataRead(CacheFileHandle *aHandle, char *aBuf,
|
|
nsresult aResult)
|
|
{
|
|
LOG(("CacheFileMetadata::OnDataRead() [this=%p, handle=%p, result=0x%08x]",
|
|
this, aHandle, aResult));
|
|
|
|
MOZ_ASSERT(mListener);
|
|
|
|
nsresult rv;
|
|
nsCOMPtr<CacheFileMetadataListener> listener;
|
|
|
|
if (NS_FAILED(aResult)) {
|
|
LOG(("CacheFileMetadata::OnDataRead() - CacheFileIOManager::Read() failed"
|
|
", creating empty metadata. [this=%p, rv=0x%08x]", this, aResult));
|
|
|
|
InitEmptyMetadata();
|
|
|
|
mListener.swap(listener);
|
|
listener->OnMetadataRead(NS_OK);
|
|
return NS_OK;
|
|
}
|
|
|
|
// check whether we have read all necessary data
|
|
uint32_t realOffset = NetworkEndian::readUint32(mBuf + mBufSize -
|
|
sizeof(uint32_t));
|
|
|
|
int64_t size = mHandle->FileSize();
|
|
MOZ_ASSERT(size != -1);
|
|
|
|
if (realOffset >= size) {
|
|
LOG(("CacheFileMetadata::OnDataRead() - Invalid realOffset, creating "
|
|
"empty metadata. [this=%p, realOffset=%u, size=%lld]", this,
|
|
realOffset, size));
|
|
|
|
InitEmptyMetadata();
|
|
|
|
mListener.swap(listener);
|
|
listener->OnMetadataRead(NS_OK);
|
|
return NS_OK;
|
|
}
|
|
|
|
uint32_t maxHashCount = size / kChunkSize;
|
|
uint32_t maxMetadataSize = CalcMetadataSize(kMaxElementsSize, maxHashCount);
|
|
if (size - realOffset > maxMetadataSize) {
|
|
LOG(("CacheFileMetadata::OnDataRead() - Invalid realOffset, metadata would "
|
|
"be too big, creating empty metadata. [this=%p, realOffset=%u, "
|
|
"maxMetadataSize=%u, size=%lld]", this, realOffset, maxMetadataSize,
|
|
size));
|
|
|
|
InitEmptyMetadata();
|
|
|
|
mListener.swap(listener);
|
|
listener->OnMetadataRead(NS_OK);
|
|
return NS_OK;
|
|
}
|
|
|
|
uint32_t usedOffset = size - mBufSize;
|
|
|
|
if (realOffset < usedOffset) {
|
|
uint32_t missing = usedOffset - realOffset;
|
|
// we need to read more data
|
|
char *newBuf = static_cast<char *>(moz_xrealloc(mBuf, mBufSize + missing));
|
|
if (!newBuf) {
|
|
LOG(("CacheFileMetadata::OnDataRead() - Error allocating %d more bytes "
|
|
"for the missing part of the metadata, creating empty metadata. "
|
|
"[this=%p]", missing, this));
|
|
|
|
InitEmptyMetadata();
|
|
|
|
mListener.swap(listener);
|
|
listener->OnMetadataRead(NS_OK);
|
|
return NS_OK;
|
|
}
|
|
|
|
mBuf = newBuf;
|
|
memmove(mBuf + missing, mBuf, mBufSize);
|
|
mBufSize += missing;
|
|
|
|
DoMemoryReport(MemoryUsage());
|
|
|
|
LOG(("CacheFileMetadata::OnDataRead() - We need to read %d more bytes to "
|
|
"have full metadata. [this=%p]", missing, this));
|
|
|
|
rv = CacheFileIOManager::Read(mHandle, realOffset, mBuf, missing, this);
|
|
if (NS_FAILED(rv)) {
|
|
LOG(("CacheFileMetadata::OnDataRead() - CacheFileIOManager::Read() "
|
|
"failed synchronously, creating empty metadata. [this=%p, "
|
|
"rv=0x%08x]", this, rv));
|
|
|
|
InitEmptyMetadata();
|
|
|
|
mListener.swap(listener);
|
|
listener->OnMetadataRead(NS_OK);
|
|
return NS_OK;
|
|
}
|
|
|
|
return NS_OK;
|
|
}
|
|
|
|
// We have all data according to offset information at the end of the entry.
|
|
// Try to parse it.
|
|
rv = ParseMetadata(realOffset, realOffset - usedOffset, true);
|
|
if (NS_FAILED(rv)) {
|
|
LOG(("CacheFileMetadata::OnDataRead() - Error parsing metadata, creating "
|
|
"empty metadata. [this=%p]", this));
|
|
InitEmptyMetadata();
|
|
} else {
|
|
// Shrink elements buffer.
|
|
mBuf = static_cast<char *>(moz_xrealloc(mBuf, mElementsSize));
|
|
mBufSize = mElementsSize;
|
|
|
|
// There is usually no or just one call to SetMetadataElement() when the
|
|
// metadata is parsed from disk. Avoid allocating power of two sized buffer
|
|
// which we do in case of newly created metadata.
|
|
mAllocExactSize = true;
|
|
}
|
|
|
|
mListener.swap(listener);
|
|
listener->OnMetadataRead(NS_OK);
|
|
|
|
return NS_OK;
|
|
}
|
|
|
|
nsresult
|
|
CacheFileMetadata::OnFileDoomed(CacheFileHandle *aHandle, nsresult aResult)
|
|
{
|
|
MOZ_CRASH("CacheFileMetadata::OnFileDoomed should not be called!");
|
|
return NS_ERROR_UNEXPECTED;
|
|
}
|
|
|
|
nsresult
|
|
CacheFileMetadata::OnEOFSet(CacheFileHandle *aHandle, nsresult aResult)
|
|
{
|
|
MOZ_CRASH("CacheFileMetadata::OnEOFSet should not be called!");
|
|
return NS_ERROR_UNEXPECTED;
|
|
}
|
|
|
|
nsresult
|
|
CacheFileMetadata::OnFileRenamed(CacheFileHandle *aHandle, nsresult aResult)
|
|
{
|
|
MOZ_CRASH("CacheFileMetadata::OnFileRenamed should not be called!");
|
|
return NS_ERROR_UNEXPECTED;
|
|
}
|
|
|
|
void
|
|
CacheFileMetadata::InitEmptyMetadata()
|
|
{
|
|
if (mBuf) {
|
|
free(mBuf);
|
|
mBuf = nullptr;
|
|
mBufSize = 0;
|
|
}
|
|
mOffset = 0;
|
|
mMetaHdr.mVersion = kCacheEntryVersion;
|
|
mMetaHdr.mFetchCount = 0;
|
|
mMetaHdr.mExpirationTime = nsICacheEntry::NO_EXPIRATION_TIME;
|
|
mMetaHdr.mKeySize = mKey.Length();
|
|
|
|
// Deliberately not touching the "kCacheEntryIsPinned" flag.
|
|
|
|
DoMemoryReport(MemoryUsage());
|
|
|
|
// We're creating a new entry. If there is any old data truncate it.
|
|
if (mHandle) {
|
|
mHandle->SetPinned(Pinned());
|
|
if (mHandle->FileExists() && mHandle->FileSize()) {
|
|
CacheFileIOManager::TruncateSeekSetEOF(mHandle, 0, 0, nullptr);
|
|
}
|
|
}
|
|
}
|
|
|
|
nsresult
|
|
CacheFileMetadata::ParseMetadata(uint32_t aMetaOffset, uint32_t aBufOffset,
|
|
bool aHaveKey)
|
|
{
|
|
LOG(("CacheFileMetadata::ParseMetadata() [this=%p, metaOffset=%d, "
|
|
"bufOffset=%d, haveKey=%u]", this, aMetaOffset, aBufOffset, aHaveKey));
|
|
|
|
nsresult rv;
|
|
|
|
uint32_t metaposOffset = mBufSize - sizeof(uint32_t);
|
|
uint32_t hashesOffset = aBufOffset + sizeof(uint32_t);
|
|
uint32_t hashCount = aMetaOffset / kChunkSize;
|
|
if (aMetaOffset % kChunkSize)
|
|
hashCount++;
|
|
uint32_t hashesLen = hashCount * sizeof(CacheHash::Hash16_t);
|
|
uint32_t hdrOffset = hashesOffset + hashesLen;
|
|
uint32_t keyOffset = hdrOffset + sizeof(CacheFileMetadataHeader);
|
|
|
|
LOG(("CacheFileMetadata::ParseMetadata() [this=%p]\n metaposOffset=%d\n "
|
|
"hashesOffset=%d\n hashCount=%d\n hashesLen=%d\n hdfOffset=%d\n "
|
|
"keyOffset=%d\n", this, metaposOffset, hashesOffset, hashCount,
|
|
hashesLen,hdrOffset, keyOffset));
|
|
|
|
if (keyOffset > metaposOffset) {
|
|
LOG(("CacheFileMetadata::ParseMetadata() - Wrong keyOffset! [this=%p]",
|
|
this));
|
|
return NS_ERROR_FILE_CORRUPTED;
|
|
}
|
|
|
|
mMetaHdr.ReadFromBuf(mBuf + hdrOffset);
|
|
|
|
if (mMetaHdr.mVersion == 1) {
|
|
// Backward compatibility before we've added flags to the header
|
|
keyOffset -= sizeof(uint32_t);
|
|
} else if (mMetaHdr.mVersion != kCacheEntryVersion) {
|
|
LOG(("CacheFileMetadata::ParseMetadata() - Not a version we understand to. "
|
|
"[version=0x%x, this=%p]", mMetaHdr.mVersion, this));
|
|
return NS_ERROR_UNEXPECTED;
|
|
}
|
|
|
|
// Update the version stored in the header to make writes
|
|
// store the header in the current version form.
|
|
mMetaHdr.mVersion = kCacheEntryVersion;
|
|
|
|
uint32_t elementsOffset = mMetaHdr.mKeySize + keyOffset + 1;
|
|
|
|
if (elementsOffset > metaposOffset) {
|
|
LOG(("CacheFileMetadata::ParseMetadata() - Wrong elementsOffset %d "
|
|
"[this=%p]", elementsOffset, this));
|
|
return NS_ERROR_FILE_CORRUPTED;
|
|
}
|
|
|
|
// check that key ends with \0
|
|
if (mBuf[elementsOffset - 1] != 0) {
|
|
LOG(("CacheFileMetadata::ParseMetadata() - Elements not null terminated. "
|
|
"[this=%p]", this));
|
|
return NS_ERROR_FILE_CORRUPTED;
|
|
}
|
|
|
|
|
|
if (!aHaveKey) {
|
|
// get the key form metadata
|
|
mKey.Assign(mBuf + keyOffset, mMetaHdr.mKeySize);
|
|
|
|
rv = ParseKey(mKey);
|
|
if (NS_FAILED(rv))
|
|
return rv;
|
|
}
|
|
else {
|
|
if (mMetaHdr.mKeySize != mKey.Length()) {
|
|
LOG(("CacheFileMetadata::ParseMetadata() - Key collision (1), key=%s "
|
|
"[this=%p]", nsCString(mBuf + keyOffset, mMetaHdr.mKeySize).get(),
|
|
this));
|
|
return NS_ERROR_FILE_CORRUPTED;
|
|
}
|
|
|
|
if (memcmp(mKey.get(), mBuf + keyOffset, mKey.Length()) != 0) {
|
|
LOG(("CacheFileMetadata::ParseMetadata() - Key collision (2), key=%s "
|
|
"[this=%p]", nsCString(mBuf + keyOffset, mMetaHdr.mKeySize).get(),
|
|
this));
|
|
return NS_ERROR_FILE_CORRUPTED;
|
|
}
|
|
}
|
|
|
|
// check metadata hash (data from hashesOffset to metaposOffset)
|
|
CacheHash::Hash32_t hashComputed, hashExpected;
|
|
hashComputed = CacheHash::Hash(mBuf + hashesOffset,
|
|
metaposOffset - hashesOffset);
|
|
hashExpected = NetworkEndian::readUint32(mBuf + aBufOffset);
|
|
|
|
if (hashComputed != hashExpected) {
|
|
LOG(("CacheFileMetadata::ParseMetadata() - Metadata hash mismatch! Hash of "
|
|
"the metadata is %x, hash in file is %x [this=%p]", hashComputed,
|
|
hashExpected, this));
|
|
return NS_ERROR_FILE_CORRUPTED;
|
|
}
|
|
|
|
// check elements
|
|
rv = CheckElements(mBuf + elementsOffset, metaposOffset - elementsOffset);
|
|
if (NS_FAILED(rv))
|
|
return rv;
|
|
|
|
if (mHandle) {
|
|
if (!mHandle->SetPinned(Pinned())) {
|
|
LOG(("CacheFileMetadata::ParseMetadata() - handle was doomed for this "
|
|
"pinning state, truncate the file [this=%p, pinned=%d]", this, Pinned()));
|
|
return NS_ERROR_FILE_CORRUPTED;
|
|
}
|
|
}
|
|
|
|
mHashArraySize = hashesLen;
|
|
mHashCount = hashCount;
|
|
if (mHashArraySize) {
|
|
mHashArray = static_cast<CacheHash::Hash16_t *>(
|
|
moz_xmalloc(mHashArraySize));
|
|
memcpy(mHashArray, mBuf + hashesOffset, mHashArraySize);
|
|
}
|
|
|
|
MarkDirty();
|
|
|
|
mElementsSize = metaposOffset - elementsOffset;
|
|
memmove(mBuf, mBuf + elementsOffset, mElementsSize);
|
|
mOffset = aMetaOffset;
|
|
|
|
DoMemoryReport(MemoryUsage());
|
|
|
|
return NS_OK;
|
|
}
|
|
|
|
nsresult
|
|
CacheFileMetadata::CheckElements(const char *aBuf, uint32_t aSize)
|
|
{
|
|
if (aSize) {
|
|
// Check if the metadata ends with a zero byte.
|
|
if (aBuf[aSize - 1] != 0) {
|
|
NS_ERROR("Metadata elements are not null terminated");
|
|
LOG(("CacheFileMetadata::CheckElements() - Elements are not null "
|
|
"terminated. [this=%p]", this));
|
|
return NS_ERROR_FILE_CORRUPTED;
|
|
}
|
|
// Check that there are an even number of zero bytes
|
|
// to match the pattern { key \0 value \0 }
|
|
bool odd = false;
|
|
for (uint32_t i = 0; i < aSize; i++) {
|
|
if (aBuf[i] == 0)
|
|
odd = !odd;
|
|
}
|
|
if (odd) {
|
|
NS_ERROR("Metadata elements are malformed");
|
|
LOG(("CacheFileMetadata::CheckElements() - Elements are malformed. "
|
|
"[this=%p]", this));
|
|
return NS_ERROR_FILE_CORRUPTED;
|
|
}
|
|
}
|
|
return NS_OK;
|
|
}
|
|
|
|
nsresult
|
|
CacheFileMetadata::EnsureBuffer(uint32_t aSize)
|
|
{
|
|
if (aSize > kMaxElementsSize) {
|
|
return NS_ERROR_FAILURE;
|
|
}
|
|
|
|
if (mBufSize < aSize) {
|
|
if (mAllocExactSize) {
|
|
// If this is not the only allocation, use power of two for following
|
|
// allocations.
|
|
mAllocExactSize = false;
|
|
} else {
|
|
// find smallest power of 2 greater than or equal to aSize
|
|
--aSize;
|
|
aSize |= aSize >> 1;
|
|
aSize |= aSize >> 2;
|
|
aSize |= aSize >> 4;
|
|
aSize |= aSize >> 8;
|
|
aSize |= aSize >> 16;
|
|
++aSize;
|
|
}
|
|
|
|
if (aSize < kInitialBufSize) {
|
|
aSize = kInitialBufSize;
|
|
}
|
|
|
|
char *newBuf = static_cast<char *>(moz_xrealloc(mBuf, aSize));
|
|
if (!newBuf) {
|
|
return NS_ERROR_OUT_OF_MEMORY;
|
|
}
|
|
mBufSize = aSize;
|
|
mBuf = newBuf;
|
|
|
|
DoMemoryReport(MemoryUsage());
|
|
}
|
|
|
|
return NS_OK;
|
|
}
|
|
|
|
nsresult
|
|
CacheFileMetadata::ParseKey(const nsACString &aKey)
|
|
{
|
|
nsCOMPtr<nsILoadContextInfo> info = CacheFileUtils::ParseKey(aKey);
|
|
NS_ENSURE_TRUE(info, NS_ERROR_FAILURE);
|
|
|
|
mAnonymous = info->IsAnonymous();
|
|
mOriginAttributes = *info->OriginAttributesPtr();
|
|
|
|
return NS_OK;
|
|
}
|
|
|
|
// Memory reporting
|
|
|
|
size_t
|
|
CacheFileMetadata::SizeOfExcludingThis(mozilla::MallocSizeOf mallocSizeOf) const
|
|
{
|
|
size_t n = 0;
|
|
// mHandle reported via CacheFileIOManager.
|
|
n += mKey.SizeOfExcludingThisIfUnshared(mallocSizeOf);
|
|
n += mallocSizeOf(mHashArray);
|
|
n += mallocSizeOf(mBuf);
|
|
n += mallocSizeOf(mWriteBuf);
|
|
// mListener is usually the owning CacheFile.
|
|
|
|
return n;
|
|
}
|
|
|
|
size_t
|
|
CacheFileMetadata::SizeOfIncludingThis(mozilla::MallocSizeOf mallocSizeOf) const
|
|
{
|
|
return mallocSizeOf(this) + SizeOfExcludingThis(mallocSizeOf);
|
|
}
|
|
|
|
} // namespace net
|
|
} // namespace mozilla
|