1 /*
2  * Copyright 2014 Google Inc. All rights reserved.
3  *
4  * Licensed under the Apache License, Version 2.0 (the "License");
5  * you may not use this file except in compliance with the License.
6  * You may obtain a copy of the License at
7  *
8  *     http://www.apache.org/licenses/LICENSE-2.0
9  *
10  * Unless required by applicable law or agreed to in writing, software
11  * distributed under the License is distributed on an "AS IS" BASIS,
12  * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13  * See the License for the specific language governing permissions and
14  * limitations under the License.
15  */
16 
17 #define IN_FRUIT_CPP_FILE 1
18 
19 #include <algorithm>
20 #include <cstdlib>
21 #include <fruit/impl/util/type_info.h>
22 #include <iostream>
23 #include <memory>
24 #include <vector>
25 
26 #include <fruit/impl/component_storage/component_storage.h>
27 #include <fruit/impl/data_structures/semistatic_graph.templates.h>
28 #include <fruit/impl/injector/injector_storage.h>
29 #include <fruit/impl/normalized_component_storage/binding_normalization.h>
30 #include <fruit/impl/normalized_component_storage/binding_normalization.templates.h>
31 
32 using std::cout;
33 using std::endl;
34 
35 using namespace fruit::impl;
36 
37 namespace fruit {
38 namespace impl {
39 
fatal(const std::string & error)40 void InjectorStorage::fatal(const std::string& error) {
41   std::cerr << "Fatal injection error: " << error << std::endl;
42   exit(1);
43 }
44 
45 // LCOV_EXCL_START
46 namespace {
47 template <typename Id, typename Value>
48 struct DummyNode {
getIdfruit::impl::__anon6c16492e0111::DummyNode49   Id getId() {
50     return Id();
51   }
isTerminalfruit::impl::__anon6c16492e0111::DummyNode52   bool isTerminal() {
53     return false;
54   }
getEdgesBeginfruit::impl::__anon6c16492e0111::DummyNode55   Id* getEdgesBegin() {
56     return nullptr;
57   }
getEdgesEndfruit::impl::__anon6c16492e0111::DummyNode58   Id* getEdgesEnd() {
59     return nullptr;
60   }
getValuefruit::impl::__anon6c16492e0111::DummyNode61   Value getValue() {
62     return Value();
63   }
64 };
65 }
66 // LCOV_EXCL_STOP
67 
InjectorStorage(ComponentStorage && component,const std::vector<TypeId,ArenaAllocator<TypeId>> & exposed_types,MemoryPool & memory_pool)68 InjectorStorage::InjectorStorage(ComponentStorage&& component,
69                                  const std::vector<TypeId, ArenaAllocator<TypeId>>& exposed_types,
70                                  MemoryPool& memory_pool)
71     : normalized_component_storage_ptr(new NormalizedComponentStorage(
72           std::move(component), exposed_types, memory_pool, NormalizedComponentStorage::WithPermanentCompression())),
73       allocator(normalized_component_storage_ptr->fixed_size_allocator_data),
74       bindings(normalized_component_storage_ptr->bindings, (DummyNode<TypeId, NormalizedBinding>*)nullptr,
75                (DummyNode<TypeId, NormalizedBinding>*)nullptr, memory_pool),
76       multibindings(std::move(normalized_component_storage_ptr->multibindings)) {
77 
78 #if FRUIT_EXTRA_DEBUG
79   bindings.checkFullyConstructed();
80 #endif
81 }
82 
InjectorStorage(const NormalizedComponentStorage & normalized_component,ComponentStorage && component,MemoryPool & memory_pool)83 InjectorStorage::InjectorStorage(const NormalizedComponentStorage& normalized_component, ComponentStorage&& component,
84                                  MemoryPool& memory_pool) {
85 
86   FixedSizeAllocator::FixedSizeAllocatorData fixed_size_allocator_data;
87   using new_bindings_vector_t = std::vector<ComponentStorageEntry, ArenaAllocator<ComponentStorageEntry>>;
88   new_bindings_vector_t new_bindings_vector = new_bindings_vector_t(ArenaAllocator<ComponentStorageEntry>(memory_pool));
89 
90   BindingNormalization::normalizeBindingsAndAddTo(std::move(component).release(), memory_pool, normalized_component,
91                                                   fixed_size_allocator_data, new_bindings_vector, multibindings);
92 
93   allocator = FixedSizeAllocator(fixed_size_allocator_data);
94 
95   bindings = Graph(normalized_component.bindings, BindingDataNodeIter{new_bindings_vector.begin()},
96                    BindingDataNodeIter{new_bindings_vector.end()}, memory_pool);
97 #if FRUIT_EXTRA_DEBUG
98   bindings.checkFullyConstructed();
99 #endif
100 }
101 
~InjectorStorage()102 InjectorStorage::~InjectorStorage() {}
103 
ensureConstructedMultibinding(NormalizedMultibindingSet & multibinding_set)104 void InjectorStorage::ensureConstructedMultibinding(NormalizedMultibindingSet& multibinding_set) {
105   for (NormalizedMultibinding& multibinding : multibinding_set.elems) {
106     if (!multibinding.is_constructed) {
107       multibinding.object = multibinding.create(*this);
108       multibinding.is_constructed = true;
109     }
110   }
111 }
112 
getMultibindings(TypeId typeInfo)113 void* InjectorStorage::getMultibindings(TypeId typeInfo) {
114   NormalizedMultibindingSet* multibinding_set = getNormalizedMultibindingSet(typeInfo);
115   if (multibinding_set == nullptr) {
116     // Not registered.
117     return nullptr;
118   }
119   return multibinding_set->get_multibindings_vector(*this).get();
120 }
121 
eagerlyInjectMultibindings()122 void InjectorStorage::eagerlyInjectMultibindings() {
123   std::lock_guard<std::recursive_mutex> lock(mutex);
124   for (auto& typeInfoInfoPair : multibindings) {
125     typeInfoInfoPair.second.get_multibindings_vector(*this);
126   }
127 }
128 
129 } // namespace impl
130 // We need a LCOV_EXCL_BR_LINE below because for some reason gcov/lcov think there's a branch there.
131 } // namespace fruit LCOV_EXCL_BR_LINE
132