Update contrib.
1 // Copyright (c) 2005-2009 Nokia Corporation and/or its subsidiary(-ies).
2 // All rights reserved.
3 // This component and the accompanying materials are made available
4 // under the terms of "Eclipse Public License v1.0"
5 // which accompanies this distribution, and is available
6 // at the URL "http://www.eclipse.org/legal/epl-v10.html".
8 // Initial Contributors:
9 // Nokia Corporation - initial contribution.
14 // Implementation of API for querying support for features on a device, and
15 // receiving notification if features are added or removed.
19 #include <e32property.h>
23 #include "featregpan.h"
24 #include "featregcmn.h"
27 * Run setup exe, wait for completion: ends only once property defined, or failed
29 * @return KErrNone on success, or system-wide error code
31 static TInt RunFeaturePropertySetupExe()
33 const TUidType setupExeUidType(KExecutableImageUid, TUid::Null(), KFeaturePropCat);
35 TInt result = setupProc.Create(KFeatRegSetupExe, KNullDesC, setupExeUidType);
36 if (result != KErrNone)
40 TRequestStatus setupStatus;
41 // request Rendezvous before Resume() to avoid race condition.
42 // Also note if request to rendezvous fails (OOM etc.) then setup exe may
43 // complete after query code, with feature property possibly undefined
44 setupProc.Rendezvous(setupStatus);
47 User::WaitForRequest(setupStatus);
48 return setupStatus.Int();
52 * Dummy feature registry implementation object - never instantiated.
55 class RFeatureRegistry::TImpl
61 * Opens connection to the Feature Registry for making non-static queries.
62 * Note all non-static queries return state at the time Open() was called;
63 * Feature Registry changes are not observed until instance closed and re-opened.
65 * @return KErrNone if successful, negative system-wide error code if fails
69 EXPORT_C TInt RFeatureRegistry::Open()
71 RProperty featureProperty;
72 TInt result = featureProperty.Attach(KFeaturePropCat, KFeaturePropKey);
73 if (result != KErrNone)
78 // read feature property header
79 TInt propertySize = 0;
80 TFeatureHeader header;
81 TPckg<TFeatureHeader> headerPckg(header);
82 TBool ranSetup = EFalse;
83 TInt setupResult = KErrNone;
86 result = featureProperty.Get(headerPckg);
87 if ((result == KErrOverflow)
88 || ((result == KErrNone) && (headerPckg.Size() >= sizeof(TFeatureHeader))))
90 if (header.IsInvalid())
96 propertySize = header.PredictedPropertySize();
97 result = KErrOverflow; // indicates successful outcome from this phase
103 if (setupResult == KErrNoMemory)
105 result = KErrNoMemory;
107 else if (setupResult == KErrCorrupt)
109 result = KErrCorrupt;
113 // must force an error return - other than KErrOverflow
114 result = KErrUnknown;
118 setupResult = RunFeaturePropertySetupExe();
122 // allocate and read property. Iterate while overflow reported
123 // in case property is republished while reading it
124 while (result == KErrOverflow)
126 // the feature property data consists of only 32-bit values
127 // allocate in TUint32 blocks to cover any alignment issues
128 TUint32 size32 = (propertySize + sizeof(TUint32) - 1) / sizeof(TUint32);
129 TUint32* propertyBuf32 = new TUint32[size32];
130 TUint8* propertyBuf = reinterpret_cast<TUint8*>(propertyBuf32);
131 if (propertyBuf == NULL)
133 result = KErrNoMemory;
136 TPtr8 propertyDes(propertyBuf, 0, propertySize);
137 result = featureProperty.Get(propertyDes);
138 if (propertyDes.Size() >= sizeof(TFeatureHeader))
140 const TFeatureHeader& headerRef = *(reinterpret_cast<const TFeatureHeader*>(propertyBuf));
141 // overflow checking for the following is already done by setup exe
142 if ((result == KErrNone) && (!headerRef.IsInvalidOrBadSize(propertyDes.Size())))
145 iImpl = reinterpret_cast<TImpl*>(propertyBuf);
148 // if it's not a valid overflow (where predicted size is indeed larger than maxsize), it's corrupt
149 if ((result != KErrOverflow) || (headerRef.PredictedPropertySize() < propertyDes.MaxSize()))
151 result = KErrCorrupt;
156 result = KErrCorrupt;
158 delete[] propertyBuf;
159 if (result != KErrOverflow)
161 result = KErrCorrupt;
166 featureProperty.Close();
167 // panic in debug mode to alert system integrators that the setup exe
168 // is absent/inaccessible or the config data is invalid in this OS
169 // configuration: a serious problem
170 __ASSERT_DEBUG(result != KErrCorrupt, Panic(EFeatRegBadConfig));
175 * Queries support for feature on the device.
176 * Non-static version requiring open instance of class.
177 * Recommended when making multiple queries.
178 * Note: returns support for feature from the time Open() was called.
180 * @param aFeatureUid Unique identifier of feature being queried
181 * @return positive value if feature is supported, zero if feature is not supported,
182 * or negative system-wide error code if could not be determined.
183 * @pre this registry instance is open
184 * @panic FeatReg EFeatRegInvalidUse if this registry instance is not open
188 EXPORT_C TInt RFeatureRegistry::QuerySupport(TUid aFeatureUid)
191 return QuerySupport(aFeatureUid, dummyInfo);
195 * Queries support for feature on the device.
196 * Non-static version requiring open instance of class.
197 * Recommended when making multiple queries.
198 * Note: returns support for feature from the time Open() was called.
200 * @param aFeatureUid Unique identifier of feature being queried
201 * @param aInfo addition status information about feature
202 * @return positive value if feature is supported, zero if feature is not supported,
203 * or negative system-wide error code if could not be determined.
204 * @pre this registry instance is open
205 * @panic FeatReg EFeatRegInvalidUse if this registry instance is not open
209 EXPORT_C TInt RFeatureRegistry::QuerySupport(TUid aFeatureUid, TUint32& aInfo)
211 __ASSERT_ALWAYS(iImpl != NULL, Panic(EFeatRegInvalidUse));
213 TFeatureHeader* header = reinterpret_cast<TFeatureHeader*>(iImpl);
214 TUint32 featureUid = aFeatureUid.iUid;
216 // try to find in feature entries first
217 TFeatureEntry* entry = reinterpret_cast<TFeatureEntry*>(header + 1);
218 if (header->iFeatureEntryCount > 0)
220 RArray<TFeatureEntry> entryArray(sizeof(TFeatureEntry), entry, header->iFeatureEntryCount);
221 TFeatureEntry searchEntry = { featureUid , 0 };
222 TInt index = entryArray.FindInUnsignedKeyOrder(searchEntry);
225 aInfo = entryArray[index].iInfo;
226 return aInfo & EStatusSupportBit;
230 // fall back to default ranges - first range to match wins
231 TFeatureRange* range = reinterpret_cast<TFeatureRange*>(entry + header->iFeatureEntryCount);
232 for (TInt i = header->iFeatureRangeCount; i > 0; --i, ++range)
234 if ((featureUid >= range->iLowUid) && (featureUid <= range->iHighUid))
236 aInfo = EStatusSupportBit;
237 return EStatusSupportBit;
241 // final default: feature not supported
247 * Closes this registry instance.
251 EXPORT_C void RFeatureRegistry::Close()
253 TUint8* propertyBuf = reinterpret_cast<TUint8*>(iImpl);
254 delete[] propertyBuf;
259 * Queries support for feature on the device.
260 * Static version recommended for single queries.
262 * @param aFeatureUid Unique identifier of feature being queried
263 * @return positive value if feature is supported, zero if feature is not supported,
264 * or negative system-wide error code if could not be determined.
268 EXPORT_C TInt RFeatureRegistry::QuerySupportS(TUid aFeatureUid)
271 return QuerySupportS(aFeatureUid, dummyInfo);
275 * Queries support for feature on the device.
276 * Static version recommended for single queries.
278 * @param aFeatureUid Unique identifier of feature being queried
279 * @param aInfo addition status information about feature
280 * @return positive value if feature is supported, zero if feature is not supported,
281 * or negative system-wide error code if could not be determined.
285 EXPORT_C TInt RFeatureRegistry::QuerySupportS(TUid aFeatureUid, TUint32& aInfo)
287 RFeatureRegistry featReg;
288 TInt result = featReg.Open();
289 if (result == KErrNone)
291 result = featReg.QuerySupport(aFeatureUid, aInfo);
298 * Implementation class allocated when RFeatureRegistryNotify is opened.
302 class RFeatureRegistryNotify::TImpl
305 RProperty iNotifyProperty;
314 * Open instance of notify object so it can be subscribed to.
316 * @return KErrNone if successful, negative system-wide error code if not
319 EXPORT_C TInt RFeatureRegistryNotify::Open()
326 TInt result = iImpl->iNotifyProperty.Attach(KFeaturePropCat, KFeaturePropKey);
327 if (result != KErrNone)
329 // must clean up memory allocated above
334 // feature property and notify property are same in current implementation
335 // hence must ensure feature property is already published to avoid false
336 // notification when it is first published (just-in-time by the next query)
337 TFeatureHeader header;
338 TPckg<TFeatureHeader> headerPckg(header);
339 result = iImpl->iNotifyProperty.Get(headerPckg);
340 if (!((result == KErrOverflow)
341 || ((result == KErrNone) && (headerPckg.Size() >= sizeof(TFeatureHeader)))))
343 RunFeaturePropertySetupExe();
345 // return fact that Attach() succeeded
350 * Issues an asynchronous request to be notified the next time the support
351 * status of any features change.
353 * To ensure that changes are not missed, always re-subscribe before
354 * querying the feature registry.
356 * If an outstanding request is cancelled through a call to Cancel(), then it
357 * completes with KErrCancel.
359 * @pre this instance of notify object must be Open and not already Subscribed to.
360 * @param aNotifyStatus The request status object to be signalled on update.
361 * @panic FeatReg EFeatRegInvalidUse if this registry notify instance is not open
364 EXPORT_C void RFeatureRegistryNotify::Subscribe(TRequestStatus &aNotifyStatus)
366 __ASSERT_ALWAYS(iImpl != NULL, Panic(EFeatRegInvalidUse));
367 iImpl->iNotifyProperty.Subscribe(aNotifyStatus);
371 * Cancels an outstanding subscription request for notification of feature registry changes.
373 * If the request has not already completed, then it completes with KErrCancel.
375 * @pre this instance of notify object must be Open
376 * @panic FeatReg EFeatRegInvalidUse if this registry notify instance is not open
379 EXPORT_C void RFeatureRegistryNotify::Cancel()
381 __ASSERT_ALWAYS(iImpl != NULL, Panic(EFeatRegInvalidUse));
382 iImpl->iNotifyProperty.Cancel();
386 * Closes the registry notify instance.
388 * Note: automatically cancels any outstanding notify subscription.
392 EXPORT_C void RFeatureRegistryNotify::Close()
396 // Have checked RProperty::Close() cancels the outstanding subscription
397 iImpl->iNotifyProperty.Close();