summaryrefslogtreecommitdiffstats
path: root/ipc/glue/SerializedStructuredCloneBuffer.cpp
blob: 9cad1ed82663945802ed2e357b4d3bbb41b4145d (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
/* -*- 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 "mozilla/ipc/SerializedStructuredCloneBuffer.h"
#include "js/StructuredClone.h"

namespace IPC {

void ParamTraits<JSStructuredCloneData>::Write(MessageWriter* aWriter,
                                               const paramType& aParam) {
  MOZ_ASSERT(!(aParam.Size() % sizeof(uint64_t)));

  // We can only construct shared memory regions up to 4Gb in size, making that
  // the maximum possible JSStructuredCloneData size.
  mozilla::CheckedUint32 size = aParam.Size();
  if (!size.isValid()) {
    aWriter->FatalError("JSStructuredCloneData over 4Gb in size");
    return;
  }
  WriteParam(aWriter, size.value());

  MessageBufferWriter bufWriter(aWriter, size.value());
  aParam.ForEachDataChunk([&](const char* aData, size_t aSize) {
    return bufWriter.WriteBytes(aData, aSize);
  });
}

bool ParamTraits<JSStructuredCloneData>::Read(MessageReader* aReader,
                                              paramType* aResult) {
  uint32_t length = 0;
  if (!ReadParam(aReader, &length)) {
    aReader->FatalError("JSStructuredCloneData length read failed");
    return false;
  }
  MOZ_ASSERT(!(length % sizeof(uint64_t)));

  // Borrowing is not suitable to use for IPC to hand out data because we often
  // want to store the data somewhere for processing after IPC has released the
  // underlying buffers.
  //
  // This deserializer previously used a mechanism to transfer ownership over
  // the underlying buffers from IPC into the JSStructuredCloneData. This was
  // removed when support for serializing over shared memory was added, as the
  // benefit for avoiding copies was limited due to it only functioning for
  // buffers under 64k in size (as larger buffers would be serialized using
  // shared memory), and it added substantial complexity to the BufferList type
  // and the IPC serialization layer due to things like buffer alignment. This
  // can be revisited in the future if it turns out to be a noticable
  // performance regression. (bug 1783242)

  mozilla::BufferList<js::SystemAllocPolicy> buffers(0, 0, 4096);
  MessageBufferReader bufReader(aReader, length);
  uint32_t read = 0;
  while (read < length) {
    size_t bufLen;
    char* buf = buffers.AllocateBytes(length - read, &bufLen);
    if (!buf) {
      // Would be nice to allow actor to control behaviour here (bug 1784307)
      NS_ABORT_OOM(length - read);
      return false;
    }
    if (!bufReader.ReadBytesInto(buf, bufLen)) {
      aReader->FatalError("JSStructuredCloneData ReadBytesInto failed");
      return false;
    }
    read += bufLen;
  }

  MOZ_ASSERT(read == length);
  *aResult = JSStructuredCloneData(
      std::move(buffers), JS::StructuredCloneScope::DifferentProcess,
      OwnTransferablePolicy::IgnoreTransferablesIfAny);
  return true;
}

}  // namespace IPC