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 0 : AggregatingModule(
34 : std::vector< rtl::Reference< Provider > > const & providers,
35 : OUString const & name):
36 0 : providers_(providers), name_(name)
37 0 : {}
38 :
39 : private:
40 0 : 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 0 : AggregatingCursor(
69 : std::vector< rtl::Reference< Provider > > const & providers,
70 : OUString const & name):
71 0 : providers_(providers), name_(name), iterator_(providers_.begin())
72 0 : { findCursor(); }
73 :
74 : private:
75 0 : 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 0 : rtl::Reference< Entity > AggregatingCursor::getNext(OUString * name) {
89 0 : while (cursor_.is()) {
90 0 : OUString n;
91 0 : rtl::Reference< Entity > ent(cursor_->getNext(&n));
92 0 : if (ent.is()) {
93 0 : if (seen_.insert(n).second) {
94 0 : if (name != 0) {
95 0 : *name = n;
96 : }
97 0 : return ent->getSort() == Entity::SORT_MODULE
98 : ? new AggregatingModule(
99 0 : providers_, (name_.isEmpty() ? name_ : name_ + ".") + n)
100 0 : : ent;
101 : }
102 : } else {
103 0 : cursor_.clear();
104 0 : findCursor();
105 : }
106 0 : }
107 0 : return rtl::Reference< Entity >();
108 : }
109 :
110 0 : void AggregatingCursor::findCursor() {
111 0 : for (; !cursor_.is() && iterator_ != providers_.end(); ++iterator_) {
112 0 : if (name_.isEmpty()) {
113 0 : cursor_ = (*iterator_)->createRootCursor();
114 : } else {
115 0 : rtl::Reference< Entity > ent((*iterator_)->findEntity(name_));
116 0 : if (ent.is() && ent->getSort() == Entity::SORT_MODULE) {
117 0 : cursor_ = static_cast< ModuleEntity * >(ent.get())->
118 0 : createCursor();
119 0 : }
120 : }
121 : }
122 0 : }
123 :
124 0 : rtl::Reference< MapCursor > AggregatingModule::createCursor() const {
125 0 : return new AggregatingCursor(providers_, name_);
126 : }
127 :
128 : }
129 :
130 0 : NoSuchFileException::~NoSuchFileException() throw () {}
131 :
132 0 : FileFormatException::~FileFormatException() throw () {}
133 :
134 11 : Entity::~Entity() throw () {}
135 :
136 0 : MapCursor::~MapCursor() throw () {}
137 :
138 0 : ModuleEntity::~ModuleEntity() throw () {}
139 :
140 11 : PublishableEntity::~PublishableEntity() throw () {}
141 :
142 0 : EnumTypeEntity::~EnumTypeEntity() throw () {}
143 :
144 0 : PlainStructTypeEntity::~PlainStructTypeEntity() throw () {}
145 :
146 4 : PolymorphicStructTypeTemplateEntity::~PolymorphicStructTypeTemplateEntity()
147 2 : throw ()
148 4 : {}
149 :
150 0 : ExceptionTypeEntity::~ExceptionTypeEntity() throw () {}
151 :
152 18 : InterfaceTypeEntity::~InterfaceTypeEntity() throw () {}
153 :
154 0 : TypedefEntity::~TypedefEntity() throw () {}
155 :
156 0 : ConstantGroupEntity::~ConstantGroupEntity() throw () {}
157 :
158 0 : SingleInterfaceBasedServiceEntity::~SingleInterfaceBasedServiceEntity() throw ()
159 0 : {}
160 :
161 0 : AccumulationBasedServiceEntity::~AccumulationBasedServiceEntity() throw () {}
162 :
163 0 : InterfaceBasedSingletonEntity::~InterfaceBasedSingletonEntity() throw () {}
164 :
165 0 : ServiceBasedSingletonEntity::~ServiceBasedSingletonEntity() throw () {}
166 :
167 3 : Provider::~Provider() throw () {}
168 :
169 3 : rtl::Reference< Provider > loadProvider(
170 : rtl::Reference< Manager > const & manager, OUString const & uri)
171 : {
172 3 : osl::DirectoryItem item;
173 3 : if (osl::DirectoryItem::get(uri, item) == osl::FileBase::E_None) {
174 3 : osl::FileStatus status(osl_FileStatus_Mask_Type);
175 6 : if (item.getFileStatus(status) == osl::FileBase::E_None
176 3 : && status.getFileType() == osl::FileStatus::Directory)
177 : {
178 0 : return new detail::SourceTreeProvider(manager, uri);
179 3 : }
180 : }
181 3 : if (uri.endsWith(".idl")) {
182 0 : return new detail::SourceFileProvider(manager, uri);
183 : }
184 : try {
185 3 : return new detail::UnoidlProvider(uri);
186 0 : } catch (FileFormatException & e) {
187 : SAL_INFO(
188 : "unoidl",
189 : "FileFormatException \"" << e.getDetail() << "\", retrying <" << uri
190 : << "> as legacy format");
191 0 : return new detail::LegacyProvider(manager, uri);
192 3 : }
193 : }
194 :
195 3 : void Manager::addProvider(rtl::Reference< Provider > const & provider) {
196 : assert(provider.is());
197 3 : osl::MutexGuard g(mutex_);
198 3 : providers_.push_back(provider);
199 3 : }
200 :
201 11 : rtl::Reference< Entity > Manager::findEntity(rtl::OUString const & name) const {
202 : //TODO: caching? (here or in cppuhelper::TypeManager?)
203 11 : osl::MutexGuard g(mutex_);
204 28 : for (std::vector< rtl::Reference< Provider > >::const_iterator i(
205 11 : providers_.begin());
206 26 : i != providers_.end(); ++i)
207 : {
208 13 : rtl::Reference< Entity > ent((*i)->findEntity(name));
209 13 : if (ent.is()) {
210 11 : return ent;
211 : }
212 2 : }
213 0 : return rtl::Reference< Entity >();
214 : }
215 :
216 0 : rtl::Reference< MapCursor > Manager::createCursor(rtl::OUString const & name)
217 : const
218 : {
219 0 : return new AggregatingCursor(providers_, name);
220 : }
221 :
222 2 : Manager::~Manager() throw () {}
223 :
224 : }
225 :
226 : /* vim:set shiftwidth=4 softtabstop=4 expandtab: */
|