Chromium Code Reviews
chromiumcodereview-hr@appspot.gserviceaccount.com (chromiumcodereview-hr) | Please choose your nickname with Settings | Help | Chromium Project | Gerrit Changes | Sign out
(224)

Side by Side Diff: tools/clang/blink_gc_plugin/CheckSingletonVisitor.cpp

Issue 2696713003: blink_gc_plugin: detect singletons with embedded ScriptWrappables.
Patch Set: non-copying iteration Created 3 years, 10 months ago
Use n/p to move between diff chunks; N/P to move between comments. Draft comments are only viewable by you.
Jump to:
View unified diff | Download patch
OLDNEW
(Empty)
1 // Copyright 2017 The Chromium Authors. All rights reserved.
2 // Use of this source code is governed by a BSD-style license that can be
3 // found in the LICENSE file.
4
5 #include "CheckSingletonVisitor.h"
6
7 #include "BlinkGCPluginOptions.h"
8 #include "Config.h"
9 #include "RecordInfo.h"
10
11 using namespace clang;
12
13 CheckSingletonVisitor::CheckSingletonVisitor() = default;
14
15 CheckSingletonVisitor::~CheckSingletonVisitor() = default;
16
17 bool CheckSingletonVisitor::IsScriptWrappable(RecordInfo* info) {
18 return Config::IsScriptWrappable(info->name()) && info->IsInBlinkNamespace();
19 }
20
21 bool CheckSingletonVisitor::InheritsScriptWrappable(CXXRecordDecl* record) {
22 RecordInfo* info = info_->cache()->Lookup(record);
23 if (!info)
24 return false;
25
26 if (IsScriptWrappable(info))
27 return true;
28
29 for (auto& base : info->GetBases())
30 if (InheritsScriptWrappable(base.first))
31 return true;
32
33 return false;
34 }
35
36 void CheckSingletonVisitor::CheckScriptWrappable(const std::string& name,
37 const Type* type) {
38 RecordInfo* info = info_->cache()->Lookup(type);
39 if (!info)
40 return;
41
42 if (IsScriptWrappable(info)) {
43 std::string context = GenerateErrorContext(name);
44 illegal_types_.push_back(std::make_pair(context, type));
45 return;
46 }
47
48 RecordInfo::TemplateArgs args;
49 if (Config::IsMember(info->name()) && info->GetTemplateArgs(1, &args)) {
50 PushType(name, args[0]);
51 return;
52 }
53 if (Config::IsWeakMember(info->name()) && info->GetTemplateArgs(1, &args)) {
54 PushType(name, args[0]);
55 return;
56 }
57 bool has_persistent_name = false;
58 if (const Type* arg_type =
59 info->GetPersistentArgumentType(has_persistent_name)) {
60 PushType(name, arg_type);
61 return;
62 }
63 if (Config::IsGCCollection(info->name()) ||
64 Config::IsWTFCollection(info->name())) {
65 if (Config::IsPersistentGCCollection(info->name()) ||
66 info->IsHeapAllocatedCollection()) {
67 size_t count = Config::CollectionDimension(info->name());
68 if (info->GetTemplateArgs(count, &args)) {
69 for (const auto& t : args)
70 PushType(name, t);
71 return;
72 }
73 }
74 }
75
76 // First check if the class inherits |blink::ScriptWrappable|,
77 // and fail early if so.
78 if (InheritsScriptWrappable(info->record())) {
79 std::string context = GenerateErrorContext(name);
80 illegal_types_.push_back(std::make_pair(context, type));
81 return;
82 }
83 for (auto& base : info->GetBases()) {
84 PushType(name, base.first->getTypeForDecl());
85 }
86 for (auto& field : info->GetFields()) {
87 CheckScriptWrappable(field.first->getNameAsString(),
88 field.first->getType().getTypePtr());
89 }
90 }
91
92 void CheckSingletonVisitor::PushType(const Type* type) {
93 if (type->isPointerType() || type->isReferenceType())
94 return;
95
96 if (already_seen_.find(type) != already_seen_.end())
97 return;
98
99 already_seen_.insert(type);
100 types_stack_.push_back(std::make_pair("", type));
101 }
102
103 void CheckSingletonVisitor::PushType(const std::string& name,
104 const Type* type) {
105 if (type->isPointerType() || type->isReferenceType())
106 return;
107
108 if (already_seen_.find(type) != already_seen_.end())
109 return;
110
111 already_seen_.insert(type);
112 types_stack_.push_back(std::make_pair(name, type));
113 }
114
115 bool CheckSingletonVisitor::CheckIfSafe(RecordInfo* info, const Type* type) {
116 info_ = info;
117 types_stack_.push_back(std::make_pair("", type));
118 while (types_stack_.size()) {
119 auto t = types_stack_.back();
120 types_stack_.pop_back();
121 if (!t.second) {
122 // We push a stack marker when extending the
123 // (field) error context - pop the context when
124 // it is encountered again.
125 error_context_.pop_back();
126 continue;
127 } else if (!t.first.empty()) {
128 error_context_.push_back(t.first);
129 types_stack_.push_back(std::make_pair(t.first, nullptr));
130 }
131 CheckScriptWrappable("", t.second);
132 }
133 return !illegal_types_.size();
134 }
135
136 std::string CheckSingletonVisitor::GenerateErrorContext(
137 const std::string& name) {
138 std::string result;
139 for (const auto& elem : error_context_) {
140 if (!result.empty())
141 result.append(" => ");
142 result.append(elem);
143 }
144 if (!name.empty()) {
145 if (!result.empty())
146 result.append(" => ");
147 result.append(name);
148 }
149 return result;
150 }
OLDNEW
« no previous file with comments | « tools/clang/blink_gc_plugin/CheckSingletonVisitor.h ('k') | tools/clang/blink_gc_plugin/CollectVisitor.h » ('j') | no next file with comments »

Powered by Google App Engine
This is Rietveld 408576698