gecko-dev/dom/file/StreamBlobImpl.cpp

168 строки
5.5 KiB
C++

/* -*- Mode: C++; tab-width: 8; indent-tabs-mode: nil; c-basic-offset: 2 -*- */
/* vim: set ts=8 sts=2 et sw=2 tw=80: */
/* 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 "EmptyBlobImpl.h"
#include "mozilla/InputStreamLengthWrapper.h"
#include "mozilla/SlicedInputStream.h"
#include "StreamBlobImpl.h"
#include "nsStreamUtils.h"
#include "nsStringStream.h"
#include "nsICloneableInputStream.h"
namespace mozilla::dom {
NS_IMPL_ISUPPORTS_INHERITED(StreamBlobImpl, BlobImpl, nsIMemoryReporter)
/* static */
already_AddRefed<StreamBlobImpl> StreamBlobImpl::Create(
already_AddRefed<nsIInputStream> aInputStream,
const nsAString& aContentType, uint64_t aLength,
const nsAString& aBlobImplType) {
nsCOMPtr<nsIInputStream> inputStream = std::move(aInputStream);
RefPtr<StreamBlobImpl> blobImplStream = new StreamBlobImpl(
inputStream.forget(), aContentType, aLength, aBlobImplType);
blobImplStream->MaybeRegisterMemoryReporter();
return blobImplStream.forget();
}
/* static */
already_AddRefed<StreamBlobImpl> StreamBlobImpl::Create(
already_AddRefed<nsIInputStream> aInputStream, const nsAString& aName,
const nsAString& aContentType, int64_t aLastModifiedDate, uint64_t aLength,
const nsAString& aBlobImplType) {
nsCOMPtr<nsIInputStream> inputStream = std::move(aInputStream);
RefPtr<StreamBlobImpl> blobImplStream =
new StreamBlobImpl(inputStream.forget(), aName, aContentType,
aLastModifiedDate, aLength, aBlobImplType);
blobImplStream->MaybeRegisterMemoryReporter();
return blobImplStream.forget();
}
StreamBlobImpl::StreamBlobImpl(already_AddRefed<nsIInputStream> aInputStream,
const nsAString& aContentType, uint64_t aLength,
const nsAString& aBlobImplType)
: BaseBlobImpl(aContentType, aLength),
mInputStream(std::move(aInputStream)),
mBlobImplType(aBlobImplType),
mIsDirectory(false),
mFileId(-1) {}
StreamBlobImpl::StreamBlobImpl(already_AddRefed<nsIInputStream> aInputStream,
const nsAString& aName,
const nsAString& aContentType,
int64_t aLastModifiedDate, uint64_t aLength,
const nsAString& aBlobImplType)
: BaseBlobImpl(aName, aContentType, aLength, aLastModifiedDate),
mInputStream(std::move(aInputStream)),
mBlobImplType(aBlobImplType),
mIsDirectory(false),
mFileId(-1) {}
StreamBlobImpl::~StreamBlobImpl() { UnregisterWeakMemoryReporter(this); }
void StreamBlobImpl::CreateInputStream(nsIInputStream** aStream,
ErrorResult& aRv) {
nsCOMPtr<nsIInputStream> clonedStream;
nsCOMPtr<nsIInputStream> replacementStream;
aRv = NS_CloneInputStream(mInputStream, getter_AddRefs(clonedStream),
getter_AddRefs(replacementStream));
if (NS_WARN_IF(aRv.Failed())) {
return;
}
if (replacementStream) {
mInputStream = std::move(replacementStream);
}
nsCOMPtr<nsIInputStream> wrappedStream =
InputStreamLengthWrapper::MaybeWrap(clonedStream.forget(), mLength);
wrappedStream.forget(aStream);
}
already_AddRefed<BlobImpl> StreamBlobImpl::CreateSlice(
uint64_t aStart, uint64_t aLength, const nsAString& aContentType,
ErrorResult& aRv) {
if (!aLength) {
RefPtr<BlobImpl> impl = new EmptyBlobImpl(aContentType);
return impl.forget();
}
nsCOMPtr<nsIInputStream> clonedStream;
nsCOMPtr<nsICloneableInputStreamWithRange> stream =
do_QueryInterface(mInputStream);
if (stream) {
aRv = stream->CloneWithRange(aStart, aLength, getter_AddRefs(clonedStream));
if (NS_WARN_IF(aRv.Failed())) {
return nullptr;
}
} else {
CreateInputStream(getter_AddRefs(clonedStream), aRv);
if (NS_WARN_IF(aRv.Failed())) {
return nullptr;
}
clonedStream =
new SlicedInputStream(clonedStream.forget(), aStart, aLength);
}
MOZ_ASSERT(clonedStream);
RefPtr<BlobImpl> impl = new StreamBlobImpl(
clonedStream.forget(), aContentType, aLength, mBlobImplType);
return impl.forget();
}
void StreamBlobImpl::MaybeRegisterMemoryReporter() {
// We report only stringInputStream.
nsCOMPtr<nsIStringInputStream> stringInputStream =
do_QueryInterface(mInputStream);
if (!stringInputStream) {
return;
}
RegisterWeakMemoryReporter(this);
}
NS_IMETHODIMP
StreamBlobImpl::CollectReports(nsIHandleReportCallback* aHandleReport,
nsISupports* aData, bool aAnonymize) {
nsCOMPtr<nsIStringInputStream> stringInputStream =
do_QueryInterface(mInputStream);
if (!stringInputStream) {
return NS_OK;
}
MOZ_COLLECT_REPORT(
"explicit/dom/memory-file-data/stream", KIND_HEAP, UNITS_BYTES,
stringInputStream->SizeOfIncludingThisIfUnshared(MallocSizeOf),
"Memory used to back a File/Blob based on an input stream.");
return NS_OK;
}
size_t StreamBlobImpl::GetAllocationSize() const {
nsCOMPtr<nsIStringInputStream> stringInputStream =
do_QueryInterface(mInputStream);
if (!stringInputStream) {
return 0;
}
return stringInputStream->SizeOfIncludingThisEvenIfShared(MallocSizeOf);
}
void StreamBlobImpl::GetBlobImplType(nsAString& aBlobImplType) const {
aBlobImplType.AssignLiteral("StreamBlobImpl[");
aBlobImplType.Append(mBlobImplType);
aBlobImplType.AppendLiteral("]");
}
} // namespace mozilla::dom