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
cc / paint / transfer_cache_fuzzer.cc [blame]
// Copyright 2024 The Chromium Authors
// Use of this source code is governed by a BSD-style license that can be
// found in the LICENSE file.
#ifdef UNSAFE_BUFFERS_BUILD
// TODO(crbug.com/351564777): Remove this and convert code to safer constructs.
#pragma allow_unsafe_buffers
#endif
#include <stddef.h>
#include <stdint.h>
#include "base/bits.h"
#include "cc/paint/raw_memory_transfer_cache_entry.h"
#include "cc/test/transfer_cache_test_helper.h"
#include "components/viz/test/test_context_provider.h"
#include "third_party/skia/include/core/SkSurface.h"
#include "third_party/skia/include/gpu/ganesh/GrDirectContext.h"
struct Environment {
Environment() { logging::SetMinLogLevel(logging::LOGGING_FATAL); }
};
// TODO(crbug.com/40266937): Implement fuzzer with Skia Graphite backend.
extern "C" int LLVMFuzzerTestOneInput(const uint8_t* data, size_t size) {
static Environment env;
// Size required for fuzzing metadata.
if (size < 2) {
return 0;
}
scoped_refptr<viz::TestContextProvider> context_provider =
viz::TestContextProvider::CreateRaster();
context_provider->BindToCurrentSequence();
cc::TransferCacheEntryType entry_type =
static_cast<cc::TransferCacheEntryType>(data[0]);
std::unique_ptr<cc::ServiceTransferCacheEntry> entry =
cc::ServiceTransferCacheEntry::Create(entry_type);
if (!entry) {
return 0;
}
#if DCHECK_IS_ON()
// Align data on debug builds. ImageTransferCacheEntry requires 16-byte
// alignment on debug builds. Note: Consume one byte becauase the first
// byte was used for `TransferCacheEntryType`
const uint8_t* aligned_data = base::bits::AlignUp(&data[1], 16);
size_t alignment_gap = aligned_data - data;
if (size < alignment_gap) {
return 0;
}
base::span<const uint8_t> span(aligned_data, size - alignment_gap);
#else
// Support memory backing to discover bugs in release builds that require
// unaligned memory.
size_t offset = data[1] % 16;
const uint8_t* unaligned_data = &data[2] + offset;
size_t unaligned_gap = unaligned_data - data;
if (size < unaligned_gap) {
return 0;
}
base::span<const uint8_t> span(unaligned_data, size - unaligned_gap);
#endif
if (!entry->Deserialize(context_provider->GrContext(),
/*graphite_recorder=*/nullptr, span)) {
return 0;
}
// TODO(enne): consider running Serialize() here to fuzz that codepath
// for bugs. However, that requires setting up a real context with
// a raster interface that supports gl operations and that has a
// TransferCacheTestHelper in it so the innards of client and service
// are accessible.
return 0;
}