Add MediaCapabilities as an Experimental Feature
[WebKit-https.git] / Source / WebCore / page / RuntimeEnabledFeatures.h
1 /*
2  * Copyright (C) 2009 Google Inc. All rights reserved.
3  * Copyright (C) 2013-2017 Apple Inc. All rights reserved.
4  *
5  * Redistribution and use in source and binary forms, with or without
6  * modification, are permitted provided that the following conditions are
7  * met:
8  *
9  *     * Redistributions of source code must retain the above copyright
10  * notice, this list of conditions and the following disclaimer.
11  *     * Redistributions in binary form must reproduce the above
12  * copyright notice, this list of conditions and the following disclaimer
13  * in the documentation and/or other materials provided with the
14  * distribution.
15  *     * Neither the name of Google Inc. nor the names of its
16  * contributors may be used to endorse or promote products derived from
17  * this software without specific prior written permission.
18  *
19  * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
20  * "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
21  * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
22  * A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
23  * OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
24  * SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
25  * LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
26  * DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
27  * THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
28  * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
29  * OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
30  */
31
32 #pragma once
33
34 #include <wtf/Forward.h>
35 #include <wtf/Noncopyable.h>
36
37 namespace WebCore {
38
39 // A class that stores static enablers for all experimental features. Note that
40 // the method names must line up with the JavaScript method they enable for code
41 // generation to work properly.
42
43 class RuntimeEnabledFeatures {
44     WTF_MAKE_NONCOPYABLE(RuntimeEnabledFeatures);
45 public:
46     void setDisplayContentsEnabled(bool isEnabled) { m_isDisplayContentsEnabled = isEnabled; }
47     bool displayContentsEnabled() const { return m_isDisplayContentsEnabled; }
48
49     void setLinkPreloadEnabled(bool isEnabled) { m_isLinkPreloadEnabled = isEnabled; }
50     bool linkPreloadEnabled() const { return m_isLinkPreloadEnabled; }
51
52     void setLinkPrefetchEnabled(bool isEnabled) { m_isLinkPrefetchEnabled = isEnabled; }
53     bool linkPrefetchEnabled() const { return m_isLinkPrefetchEnabled; }
54
55     void setMediaPreloadingEnabled(bool isEnabled) { m_isMediaPreloadingEnabled = isEnabled; }
56     bool mediaPreloadingEnabled() const { return m_isMediaPreloadingEnabled; }
57
58     void setResourceTimingEnabled(bool isEnabled) { m_isResourceTimingEnabled = isEnabled; }
59     bool resourceTimingEnabled() const { return m_isResourceTimingEnabled; }
60
61     void setUserTimingEnabled(bool isEnabled) { m_isUserTimingEnabled = isEnabled; }
62     bool userTimingEnabled() const { return m_isUserTimingEnabled; }
63
64     bool performanceTimelineEnabled() const { return resourceTimingEnabled() || userTimingEnabled(); }
65
66     void setShadowDOMEnabled(bool isEnabled) { m_isShadowDOMEnabled = isEnabled; }
67     bool shadowDOMEnabled() const { return m_isShadowDOMEnabled; }
68
69     void setInputEventsEnabled(bool isEnabled) { m_inputEventsEnabled = isEnabled; }
70     bool inputEventsEnabled() const { return m_inputEventsEnabled; }
71
72     void setInteractiveFormValidationEnabled(bool isEnabled) { m_isInteractiveFormValidationEnabled = isEnabled; }
73     bool interactiveFormValidationEnabled() const { return m_isInteractiveFormValidationEnabled; }
74
75     void setCustomElementsEnabled(bool areEnabled) { m_areCustomElementsEnabled = areEnabled; }
76     bool customElementsEnabled() const { return m_areCustomElementsEnabled; }
77
78     void setMenuItemElementEnabled(bool isEnabled) { m_isMenuItemElementEnabled = isEnabled; }
79     bool menuItemElementEnabled() const { return m_isMenuItemElementEnabled; }
80     
81     void setDirectoryUploadEnabled(bool isEnabled) { m_isDirectoryUploadEnabled = isEnabled; }
82     bool directoryUploadEnabled() const { return m_isDirectoryUploadEnabled; }
83
84     void setDataTransferItemsEnabled(bool areEnabled) { m_areDataTransferItemsEnabled = areEnabled; }
85     bool dataTransferItemsEnabled() const { return m_areDataTransferItemsEnabled; }
86
87     void setCustomPasteboardDataEnabled(bool isEnabled) { m_isCustomPasteboardDataEnabled = isEnabled; }
88     bool customPasteboardDataEnabled() const { return m_isCustomPasteboardDataEnabled; }
89     
90     void setWebShareEnabled(bool isEnabled) { m_isWebShareEnabled = isEnabled; }
91     bool webShareEnabled() const { return m_isWebShareEnabled; }
92
93 #if ENABLE(ATTACHMENT_ELEMENT)
94     void setAttachmentElementEnabled(bool areEnabled) { m_isAttachmentElementEnabled = areEnabled; }
95     bool attachmentElementEnabled() const { return m_isAttachmentElementEnabled; }
96 #endif
97
98     void setModernMediaControlsEnabled(bool areEnabled) { m_areModernMediaControlsEnabled = areEnabled; }
99     bool modernMediaControlsEnabled() const { return m_areModernMediaControlsEnabled; }
100
101     void setWebAuthenticationEnabled(bool isEnabled) { m_isWebAuthenticationEnabled = isEnabled; }
102     bool webAuthenticationEnabled() const { return m_isWebAuthenticationEnabled; }
103
104     void setIsSecureContextAttributeEnabled(bool isEnabled) { m_isSecureContextAttributeEnabled = isEnabled; }
105     bool isSecureContextAttributeEnabled() const { return m_isSecureContextAttributeEnabled; }
106
107 #if ENABLE(INDEXED_DATABASE_IN_WORKERS)
108     void setIndexedDBWorkersEnabled(bool isEnabled) { m_isIndexedDBWorkersEnabled = isEnabled; }
109     bool indexedDBWorkersEnabled() const { return m_isIndexedDBWorkersEnabled; }
110 #endif
111
112 #if ENABLE(MEDIA_STREAM)
113     bool mediaDevicesEnabled() const { return m_isMediaDevicesEnabled; }
114     void setMediaDevicesEnabled(bool isEnabled) { m_isMediaDevicesEnabled = isEnabled; }
115     bool mediaStreamEnabled() const { return m_isMediaStreamEnabled; }
116     void setMediaStreamEnabled(bool isEnabled) { m_isMediaStreamEnabled = isEnabled; }
117     bool screenCaptureEnabled() const { return m_isScreenCaptureEnabled; }
118     void setScreenCaptureEnabled(bool isEnabled) { m_isScreenCaptureEnabled = isEnabled; }
119 #endif
120
121 #if ENABLE(WEB_RTC)
122     bool webRTCUnifiedPlanEnabled() const { return m_isWebRTCUnifiedPlanEnabled; }
123     void setWebRTCUnifiedPlanEnabled(bool isEnabled) { m_isWebRTCUnifiedPlanEnabled = isEnabled; }
124     bool peerConnectionEnabled() const { return m_isPeerConnectionEnabled; }
125     void setPeerConnectionEnabled(bool isEnabled) { m_isPeerConnectionEnabled = isEnabled; }
126     bool mdnsICECandidatesEnabled() const { return m_mdnsICECandidatesEnabled; }
127     void setMDNSICECandidatesEnabled(bool isEnabled) { m_mdnsICECandidatesEnabled = isEnabled; }
128 #endif
129
130 #if ENABLE(LEGACY_CSS_VENDOR_PREFIXES)
131     void setLegacyCSSVendorPrefixesEnabled(bool isEnabled) { m_isLegacyCSSVendorPrefixesEnabled = isEnabled; }
132     bool legacyCSSVendorPrefixesEnabled() const { return m_isLegacyCSSVendorPrefixesEnabled; }
133 #endif
134
135 #if ENABLE(INPUT_TYPE_DATE)
136     bool inputTypeDateEnabled() const { return m_isInputTypeDateEnabled; }
137     void setInputTypeDateEnabled(bool isEnabled) { m_isInputTypeDateEnabled = isEnabled; }
138 #endif
139
140 #if ENABLE(INPUT_TYPE_DATETIME_INCOMPLETE)
141     bool inputTypeDateTimeEnabled() const { return m_isInputTypeDateTimeEnabled; }
142     void setInputTypeDateTimeEnabled(bool isEnabled) { m_isInputTypeDateTimeEnabled = isEnabled; }
143 #endif
144
145 #if ENABLE(INPUT_TYPE_DATETIMELOCAL)
146     bool inputTypeDateTimeLocalEnabled() const { return m_isInputTypeDateTimeLocalEnabled; }
147     void setInputTypeDateTimeLocalEnabled(bool isEnabled) { m_isInputTypeDateTimeLocalEnabled = isEnabled; }
148 #endif
149
150 #if ENABLE(INPUT_TYPE_MONTH)
151     bool inputTypeMonthEnabled() const { return m_isInputTypeMonthEnabled; }
152     void setInputTypeMonthEnabled(bool isEnabled) { m_isInputTypeMonthEnabled = isEnabled; }
153 #endif
154
155 #if ENABLE(INPUT_TYPE_TIME)
156     bool inputTypeTimeEnabled() const { return m_isInputTypeTimeEnabled; }
157     void setInputTypeTimeEnabled(bool isEnabled) { m_isInputTypeTimeEnabled = isEnabled; }
158 #endif
159
160 #if ENABLE(INPUT_TYPE_WEEK)
161     bool inputTypeWeekEnabled() const { return m_isInputTypeWeekEnabled; }
162     void setInputTypeWeekEnabled(bool isEnabled) { m_isInputTypeWeekEnabled = isEnabled; }
163 #endif
164
165 #if ENABLE(GAMEPAD)
166     void setGamepadsEnabled(bool areEnabled) { m_areGamepadsEnabled = areEnabled; }
167     bool gamepadsEnabled() const { return m_areGamepadsEnabled; }
168 #endif
169
170 #if ENABLE(CSS_ANIMATIONS_LEVEL_2)
171     // FIXME: This is not used.
172     void setAnimationTriggersEnabled(bool areEnabled) { m_areAnimationTriggersEnabled = areEnabled; }
173     bool animationTriggersEnabled() const { return m_areAnimationTriggersEnabled; }
174 #endif
175
176     void setWebAnimationsEnabled(bool areEnabled) { m_areWebAnimationsEnabled = areEnabled; }
177     bool webAnimationsEnabled() const { return m_areWebAnimationsEnabled; }
178
179     void setWebAnimationsCSSIntegrationEnabled(bool isEnabled) { m_isWebAnimationsCSSIntegrationEnabled = isEnabled; }
180     bool webAnimationsCSSIntegrationEnabled() const { return m_areWebAnimationsEnabled && m_isWebAnimationsCSSIntegrationEnabled; }
181
182 #if ENABLE(WEBGL2)
183     void setWebGL2Enabled(bool isEnabled) { m_isWebGL2Enabled = isEnabled; }
184     bool webGL2Enabled() const { return m_isWebGL2Enabled; }
185 #endif
186
187 #if ENABLE(WEBGPU)
188     void setWebGPUEnabled(bool isEnabled) { m_isWebGPUEnabled = isEnabled; }
189     bool webGPUEnabled() const { return m_isWebGPUEnabled; }
190 #endif
191
192     void setImageBitmapOffscreenCanvasEnabled(bool isEnabled) { m_isImageBitmapOffscreenCanvasEnabled = isEnabled; }
193     bool imageBitmapOffscreenCanvasEnabled() const { return m_isImageBitmapOffscreenCanvasEnabled; }
194
195     void setCacheAPIEnabled(bool isEnabled) { m_isCacheAPIEnabled = isEnabled; }
196     bool cacheAPIEnabled() const { return m_isCacheAPIEnabled; }
197
198     void setFetchAPIEnabled(bool isEnabled) { m_isFetchAPIEnabled = isEnabled; }
199     bool fetchAPIEnabled() const { return m_isFetchAPIEnabled; }
200
201     void setWebSocketEnabled(bool isEnabled) { m_isWebSocketEnabled = isEnabled; }
202     bool webSocketEnabled() const { return m_isWebSocketEnabled; }
203
204 #if ENABLE(STREAMS_API)
205     void setReadableByteStreamAPIEnabled(bool isEnabled) { m_isReadableByteStreamAPIEnabled = isEnabled; }
206     bool readableByteStreamAPIEnabled() const { return m_isReadableByteStreamAPIEnabled; }
207     void setWritableStreamAPIEnabled(bool isEnabled) { m_isWritableStreamAPIEnabled = isEnabled; }
208     bool writableStreamAPIEnabled() const { return m_isWritableStreamAPIEnabled; }
209 #endif
210     
211 #if ENABLE(DOWNLOAD_ATTRIBUTE)
212     void setDownloadAttributeEnabled(bool isEnabled) { m_isDownloadAttributeEnabled = isEnabled; }
213     bool downloadAttributeEnabled() const { return m_isDownloadAttributeEnabled; }
214 #endif
215
216 #if ENABLE(INTERSECTION_OBSERVER)
217     void setIntersectionObserverEnabled(bool isEnabled) { m_intersectionObserverEnabled = isEnabled; }
218     bool intersectionObserverEnabled() const { return m_intersectionObserverEnabled; }
219 #endif
220
221 #if ENABLE(ENCRYPTED_MEDIA)
222     void setEncryptedMediaAPIEnabled(bool isEnabled) { m_encryptedMediaAPIEnabled = isEnabled; }
223     bool encryptedMediaAPIEnabled() const { return m_encryptedMediaAPIEnabled; }
224 #endif
225
226 #if ENABLE(LEGACY_ENCRYPTED_MEDIA)
227     void setLegacyEncryptedMediaAPIEnabled(bool isEnabled) { m_legacyEncryptedMediaAPIEnabled = isEnabled; }
228     bool legacyEncryptedMediaAPIEnabled() const { return m_legacyEncryptedMediaAPIEnabled; }
229 #endif
230
231 #if ENABLE(SERVICE_WORKER)
232     bool serviceWorkerEnabled() const { return m_serviceWorkerEnabled; }
233     void setServiceWorkerEnabled(bool isEnabled) { m_serviceWorkerEnabled = isEnabled; }
234 #endif
235
236     bool fetchAPIKeepAliveEnabled() const { return m_fetchAPIKeepAliveEnabled; }
237     void setFetchAPIKeepAliveEnabled(bool isEnabled) { m_fetchAPIKeepAliveEnabled = isEnabled; }
238
239     bool spectreGadgetsEnabled() const;
240
241     void setInspectorAdditionsEnabled(bool isEnabled) { m_inspectorAdditionsEnabled = isEnabled; }
242     bool inspectorAdditionsEnabled() const { return m_inspectorAdditionsEnabled; }
243
244     void setWebVREnabled(bool isEnabled) { m_webVREnabled = isEnabled; }
245     bool webVREnabled() const { return m_webVREnabled; }
246
247     void setAccessibilityObjectModelEnabled(bool isEnabled) { m_accessibilityObjectModelEnabled = isEnabled; }
248     bool accessibilityObjectModelEnabled() const { return m_accessibilityObjectModelEnabled; }
249     
250     void setAriaReflectionEnabled(bool isEnabled) { m_ariaReflectionEnabled = isEnabled; }
251     bool ariaReflectionEnabled() const { return m_ariaReflectionEnabled; }
252
253     void setResourceLoadStatisticsDebugMode(bool isEnabled) { m_resourceLoadStatisticsDebugMode = isEnabled; }
254     bool resourceLoadStatisticsDebugMode() const { return m_resourceLoadStatisticsDebugMode; }
255
256     void setRestrictedHTTPResponseAccess(bool isEnabled) { m_isRestrictedHTTPResponseAccess = isEnabled; }
257     bool restrictedHTTPResponseAccess() const { return m_isRestrictedHTTPResponseAccess; }
258
259     void setCrossOriginResourcePolicyEnabled(bool isEnabled) { m_crossOriginResourcePolicyEnabled = isEnabled; }
260     bool crossOriginResourcePolicyEnabled() const { return m_crossOriginResourcePolicyEnabled; }
261     
262     void setWebGLCompressedTextureASTCSupportEnabled(bool isEnabled) { m_isWebGLCompressedTextureASTCSupportEnabled = isEnabled; }
263     bool webGLCompressedTextureASTCSupportEnabled() const { return m_isWebGLCompressedTextureASTCSupportEnabled; }
264
265     void setStorageAccessPromptsEnabled(bool isEnabled)  { m_promptForStorageAccessAPIEnabled = isEnabled; }
266     bool storageAccessPromptsEnabled() const { return m_promptForStorageAccessAPIEnabled; }
267
268     void setServerTimingEnabled(bool isEnabled) { m_isServerTimingEnabled = isEnabled; }
269     bool serverTimingEnabled() const { return m_isServerTimingEnabled; }
270
271     void setExperimentalPlugInSandboxProfilesEnabled(bool isEnabled) { m_experimentalPlugInSandboxProfilesEnabled = isEnabled; }
272     bool experimentalPlugInSandboxProfilesEnabled() const { return m_experimentalPlugInSandboxProfilesEnabled; }
273
274     void setDisabledAdaptationsMetaTagEnabled(bool isEnabled) { m_disabledAdaptationsMetaTagEnabled = isEnabled; }
275     bool disabledAdaptationsMetaTagEnabled() const { return m_disabledAdaptationsMetaTagEnabled; }
276
277 #if USE(SYSTEM_PREVIEW)
278     void setSystemPreviewEnabled(bool isEnabled) { m_systemPreviewEnabled = isEnabled; }
279     bool systemPreviewEnabled() const { return m_systemPreviewEnabled; }
280 #endif
281
282     void setAttrStyleEnabled(bool isEnabled) { m_attrStyleEnabled = isEnabled; }
283     bool attrStyleEnabled() const { return m_attrStyleEnabled; }
284
285     WEBCORE_EXPORT static RuntimeEnabledFeatures& sharedFeatures();
286
287 private:
288     // Never instantiate.
289     RuntimeEnabledFeatures();
290
291     bool m_areModernMediaControlsEnabled { false };
292     bool m_isLinkPreloadEnabled { true };
293     bool m_isLinkPrefetchEnabled { false };
294     bool m_isMediaPreloadingEnabled { false };
295     bool m_isResourceTimingEnabled { false };
296     bool m_isUserTimingEnabled { false };
297     bool m_isInteractiveFormValidationEnabled { false };
298     bool m_isWebAuthenticationEnabled { false };
299     bool m_isSecureContextAttributeEnabled { false };
300     bool m_isDisplayContentsEnabled { true };
301     bool m_isShadowDOMEnabled { true };
302     bool m_areCustomElementsEnabled { true };
303     bool m_isMenuItemElementEnabled { false };
304     bool m_isDirectoryUploadEnabled { false };
305     bool m_areDataTransferItemsEnabled { false };
306     bool m_isCustomPasteboardDataEnabled { false };
307     bool m_isWebShareEnabled { false };
308     bool m_inputEventsEnabled { true };
309
310 #if ENABLE(ATTACHMENT_ELEMENT)
311     bool m_isAttachmentElementEnabled { false };
312 #endif
313
314 #if ENABLE(INDEXED_DATABASE_IN_WORKERS)
315     bool m_isIndexedDBWorkersEnabled { true };
316 #endif
317
318 #if ENABLE(MEDIA_STREAM)
319     bool m_isMediaDevicesEnabled { false };
320     bool m_isMediaStreamEnabled { true };
321     bool m_isScreenCaptureEnabled { false };
322 #endif
323
324 #if ENABLE(WEB_RTC)
325     bool m_isWebRTCUnifiedPlanEnabled { true };
326     bool m_isPeerConnectionEnabled { true };
327     bool m_mdnsICECandidatesEnabled { false };
328 #endif
329
330 #if ENABLE(LEGACY_CSS_VENDOR_PREFIXES)
331     bool m_isLegacyCSSVendorPrefixesEnabled { false };
332 #endif
333
334 #if ENABLE(INPUT_TYPE_DATE)
335     bool m_isInputTypeDateEnabled { true };
336 #endif
337
338 #if ENABLE(INPUT_TYPE_DATETIME_INCOMPLETE)
339     bool m_isInputTypeDateTimeEnabled { false };
340 #endif
341
342 #if ENABLE(INPUT_TYPE_DATETIMELOCAL)
343     bool m_isInputTypeDateTimeLocalEnabled { true };
344 #endif
345
346 #if ENABLE(INPUT_TYPE_MONTH)
347     bool m_isInputTypeMonthEnabled { true };
348 #endif
349
350 #if ENABLE(INPUT_TYPE_TIME)
351     bool m_isInputTypeTimeEnabled { true };
352 #endif
353
354 #if ENABLE(INPUT_TYPE_WEEK)
355     bool m_isInputTypeWeekEnabled { true };
356 #endif
357
358 #if ENABLE(GAMEPAD)
359     bool m_areGamepadsEnabled { false };
360 #endif
361
362 #if ENABLE(CSS_ANIMATIONS_LEVEL_2)
363     bool m_areAnimationTriggersEnabled { false };
364 #endif
365
366 #if ENABLE(STREAMS_API)
367     bool m_isReadableByteStreamAPIEnabled { false };
368     bool m_isWritableStreamAPIEnabled { false };
369 #endif
370
371     bool m_areWebAnimationsEnabled { true };
372     bool m_isWebAnimationsCSSIntegrationEnabled { false };
373
374 #if ENABLE(WEBGL2)
375     bool m_isWebGL2Enabled { false };
376 #endif
377
378 #if ENABLE(WEBGPU)
379     bool m_isWebGPUEnabled { false };
380 #endif
381
382     bool m_isImageBitmapOffscreenCanvasEnabled { true };
383     bool m_isCacheAPIEnabled { false };
384     bool m_isFetchAPIEnabled { true };
385
386     bool m_isWebSocketEnabled { true };
387
388 #if ENABLE(DOWNLOAD_ATTRIBUTE)
389     bool m_isDownloadAttributeEnabled { false };
390 #endif
391
392 #if ENABLE(ENCRYPTED_MEDIA)
393     bool m_encryptedMediaAPIEnabled { false };
394 #endif
395
396 #if ENABLE(LEGACY_ENCRYPTED_MEDIA)
397     bool m_legacyEncryptedMediaAPIEnabled { false };
398 #endif
399
400 #if ENABLE(INTERSECTION_OBSERVER)
401     bool m_intersectionObserverEnabled { false };
402 #endif
403
404 #if ENABLE(SERVICE_WORKER)
405     bool m_serviceWorkerEnabled { false };
406 #endif
407
408     bool m_fetchAPIKeepAliveEnabled { false };
409     bool m_inspectorAdditionsEnabled { false };
410     bool m_webVREnabled { false };
411     bool m_accessibilityObjectModelEnabled { false };
412     bool m_ariaReflectionEnabled { true };
413     bool m_resourceLoadStatisticsDebugMode { false };
414     bool m_isRestrictedHTTPResponseAccess { true };
415     bool m_crossOriginResourcePolicyEnabled { true };
416     bool m_isWebGLCompressedTextureASTCSupportEnabled { false };
417     bool m_promptForStorageAccessAPIEnabled { false };
418     bool m_isServerTimingEnabled { false };
419     bool m_experimentalPlugInSandboxProfilesEnabled { false };
420     bool m_disabledAdaptationsMetaTagEnabled { false };
421
422 #if USE(SYSTEM_PREVIEW)
423     bool m_systemPreviewEnabled { false };
424 #endif
425
426     bool m_attrStyleEnabled { false };
427
428     friend class WTF::NeverDestroyed<RuntimeEnabledFeatures>;
429 };
430
431 } // namespace WebCore