Line data Source code
1 : /* -*- Mode: C++; tab-width: 4; indent-tabs-mode: nil; c-basic-offset: 4 -*- */
2 : /*
3 : * This file is part of the LibreOffice project.
4 : *
5 : * This Source Code Form is subject to the terms of the Mozilla Public
6 : * License, v. 2.0. If a copy of the MPL was not distributed with this
7 : * file, You can obtain one at http://mozilla.org/MPL/2.0/.
8 : */
9 :
10 : #include "sal/config.h"
11 :
12 : #include <set>
13 : #include <vector>
14 :
15 : #include "osl/file.h"
16 : #include "osl/file.hxx"
17 : #include "osl/mutex.hxx"
18 : #include "rtl/ref.hxx"
19 : #include "rtl/ustring.hxx"
20 : #include "unoidl/unoidl.hxx"
21 :
22 : #include "legacyprovider.hxx"
23 : #include "sourcefileprovider.hxx"
24 : #include "sourcetreeprovider.hxx"
25 : #include "unoidlprovider.hxx"
26 :
27 : namespace unoidl {
28 :
29 : namespace {
30 :
31 : class AggregatingModule: public ModuleEntity {
32 : public:
33 622 : AggregatingModule(
34 : std::vector< rtl::Reference< Provider > > const & providers,
35 : OUString const & name):
36 622 : providers_(providers), name_(name)
37 622 : {}
38 :
39 : private:
40 1244 : virtual ~AggregatingModule() throw () {}
41 :
42 : virtual std::vector< OUString > getMemberNames() const SAL_OVERRIDE;
43 :
44 : virtual rtl::Reference< MapCursor > createCursor() const SAL_OVERRIDE;
45 :
46 : std::vector< rtl::Reference< Provider > > providers_;
47 : OUString name_;
48 : };
49 :
50 0 : std::vector< OUString > AggregatingModule::getMemberNames() const {
51 0 : std::set< OUString > names;
52 0 : for (std::vector< rtl::Reference< Provider > >::const_iterator i(
53 0 : providers_.begin());
54 0 : i != providers_.end(); ++i)
55 : {
56 0 : rtl::Reference< Entity > ent((*i)->findEntity(name_));
57 0 : if (ent.is() && ent->getSort() == Entity::SORT_MODULE) {
58 : std::vector< OUString > ns(
59 0 : static_cast< ModuleEntity * >(ent.get())->getMemberNames());
60 0 : names.insert(ns.begin(), ns.end());
61 : }
62 0 : }
63 0 : return std::vector< OUString >(names.begin(), names.end());
64 : }
65 :
66 : class AggregatingCursor: public MapCursor {
67 : public:
68 492 : AggregatingCursor(
69 : std::vector< rtl::Reference< Provider > > const & providers,
70 : OUString const & name):
71 492 : providers_(providers), name_(name), iterator_(providers_.begin())
72 492 : { findCursor(); }
73 :
74 : private:
75 984 : virtual ~AggregatingCursor() throw () {}
76 :
77 : virtual rtl::Reference< Entity > getNext(OUString * name) SAL_OVERRIDE;
78 :
79 : void findCursor();
80 :
81 : std::vector< rtl::Reference< Provider > > providers_;
82 : OUString name_;
83 : std::vector< rtl::Reference< Provider > >::iterator iterator_;
84 : rtl::Reference< MapCursor > cursor_;
85 : std::set< OUString > seen_;
86 : };
87 :
88 18508 : rtl::Reference< Entity > AggregatingCursor::getNext(OUString * name) {
89 37785 : while (cursor_.is()) {
90 18785 : OUString n;
91 19554 : rtl::Reference< Entity > ent(cursor_->getNext(&n));
92 18785 : if (ent.is()) {
93 18229 : if (seen_.insert(n).second) {
94 18016 : if (name != 0) {
95 18016 : *name = n;
96 : }
97 18016 : return ent->getSort() == Entity::SORT_MODULE
98 : ? new AggregatingModule(
99 19857 : providers_, (name_.isEmpty() ? name_ : name_ + ".") + n)
100 36679 : : ent;
101 : }
102 : } else {
103 556 : cursor_.clear();
104 556 : findCursor();
105 : }
106 769 : }
107 492 : return rtl::Reference< Entity >();
108 : }
109 :
110 1048 : void AggregatingCursor::findCursor() {
111 2221 : for (; !cursor_.is() && iterator_ != providers_.end(); ++iterator_) {
112 1173 : if (name_.isEmpty()) {
113 31 : cursor_ = (*iterator_)->createRootCursor();
114 : } else {
115 1142 : rtl::Reference< Entity > ent((*iterator_)->findEntity(name_));
116 1142 : if (ent.is() && ent->getSort() == Entity::SORT_MODULE) {
117 1575 : cursor_ = static_cast< ModuleEntity * >(ent.get())->
118 1050 : createCursor();
119 1142 : }
120 : }
121 : }
122 1048 : }
123 :
124 18 : rtl::Reference< MapCursor > AggregatingModule::createCursor() const {
125 18 : return new AggregatingCursor(providers_, name_);
126 : }
127 :
128 : }
129 :
130 0 : NoSuchFileException::~NoSuchFileException() throw () {}
131 :
132 0 : FileFormatException::~FileFormatException() throw () {}
133 :
134 857619 : Entity::~Entity() throw () {}
135 :
136 1510 : MapCursor::~MapCursor() throw () {}
137 :
138 3533 : ModuleEntity::~ModuleEntity() throw () {}
139 :
140 854086 : PublishableEntity::~PublishableEntity() throw () {}
141 :
142 30110 : EnumTypeEntity::~EnumTypeEntity() throw () {}
143 :
144 95290 : PlainStructTypeEntity::~PlainStructTypeEntity() throw () {}
145 :
146 2638 : PolymorphicStructTypeTemplateEntity::~PolymorphicStructTypeTemplateEntity()
147 1319 : throw ()
148 2638 : {}
149 :
150 135612 : ExceptionTypeEntity::~ExceptionTypeEntity() throw () {}
151 :
152 1364900 : InterfaceTypeEntity::~InterfaceTypeEntity() throw () {}
153 :
154 8120 : TypedefEntity::~TypedefEntity() throw () {}
155 :
156 29098 : ConstantGroupEntity::~ConstantGroupEntity() throw () {}
157 :
158 12576 : SingleInterfaceBasedServiceEntity::~SingleInterfaceBasedServiceEntity() throw ()
159 12576 : {}
160 :
161 27448 : AccumulationBasedServiceEntity::~AccumulationBasedServiceEntity() throw () {}
162 :
163 2362 : InterfaceBasedSingletonEntity::~InterfaceBasedSingletonEntity() throw () {}
164 :
165 18 : ServiceBasedSingletonEntity::~ServiceBasedSingletonEntity() throw () {}
166 :
167 1551 : Provider::~Provider() throw () {}
168 :
169 1640 : rtl::Reference< Provider > Manager::addProvider(OUString const & uri) {
170 1640 : rtl::Reference< Provider > p(loadProvider(uri));
171 : assert(p.is());
172 : {
173 1455 : osl::MutexGuard g(mutex_);
174 1455 : providers_.push_back(p);
175 : }
176 1455 : return p;
177 : }
178 :
179 856641 : rtl::Reference< Entity > Manager::findEntity(rtl::OUString const & name) const {
180 : //TODO: caching? (here or in cppuhelper::TypeManager?)
181 856641 : osl::MutexGuard g(mutex_);
182 4167933 : for (std::vector< rtl::Reference< Provider > >::const_iterator i(
183 856641 : providers_.begin());
184 3349716 : i != providers_.end(); ++i)
185 : {
186 1636358 : rtl::Reference< Entity > ent((*i)->findEntity(name));
187 1636358 : if (ent.is()) {
188 818141 : return ent;
189 : }
190 818217 : }
191 38500 : return rtl::Reference< Entity >();
192 : }
193 :
194 474 : rtl::Reference< MapCursor > Manager::createCursor(rtl::OUString const & name)
195 : const
196 : {
197 474 : return new AggregatingCursor(providers_, name);
198 : }
199 :
200 1674 : Manager::~Manager() throw () {}
201 :
202 1640 : rtl::Reference< Provider > Manager::loadProvider(OUString const & uri) {
203 1640 : osl::DirectoryItem item;
204 1640 : if (osl::DirectoryItem::get(uri, item) == osl::FileBase::E_None) {
205 1640 : osl::FileStatus status(osl_FileStatus_Mask_Type);
206 3280 : if (item.getFileStatus(status) == osl::FileBase::E_None
207 1640 : && status.getFileType() == osl::FileStatus::Directory)
208 : {
209 334 : return new detail::SourceTreeProvider(*this, uri);
210 1306 : }
211 : }
212 1306 : if (uri.endsWith(".idl")) {
213 521 : return new detail::SourceFileProvider(this, uri);
214 : }
215 : try {
216 970 : return new detail::UnoidlProvider(uri);
217 0 : } catch (FileFormatException & e) {
218 : SAL_INFO(
219 : "unoidl",
220 : "FileFormatException \"" << e.getDetail() << "\", retrying <" << uri
221 : << "> as legacy format");
222 0 : return new detail::LegacyProvider(*this, uri);
223 1640 : }
224 : }
225 :
226 : }
227 :
228 : /* vim:set shiftwidth=4 softtabstop=4 expandtab: */
|