blob: 99e85036695a3a81bb06dc9d2fd3afbf0f718dfb [file] [log] [blame]
Shunkai Yao51202502022-12-12 06:11:46 +00001/*
2 * Copyright (C) 2022 The Android Open Source Project
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#pragma once
18
Mikhail Naganov5b1eed12023-01-25 11:29:11 -080019#include <map>
20#include <set>
21#include <vector>
22
Mikhail Naganove7a26ad2023-05-25 17:36:48 -070023#include <aidl/android/media/audio/IHalAdapterVendorExtension.h>
Mikhail Naganov31d46652023-01-10 18:29:25 +000024#include <aidl/android/hardware/audio/core/BpModule.h>
Mikhail Naganovf56ce782023-01-25 11:29:11 -080025#include <aidl/android/hardware/audio/core/sounddose/BpSoundDose.h>
Mikhail Naganovdfd594e2023-02-08 16:59:41 -080026#include <android-base/thread_annotations.h>
Shunkai Yao51202502022-12-12 06:11:46 +000027#include <media/audiohal/DeviceHalInterface.h>
28#include <media/audiohal/EffectHalInterface.h>
Mikhail Naganove93a0862023-03-15 17:06:59 -070029#include <media/audiohal/StreamHalInterface.h>
Shunkai Yao51202502022-12-12 06:11:46 +000030
Mikhail Naganov31d46652023-01-10 18:29:25 +000031#include "ConversionHelperAidl.h"
Shunkai Yao51202502022-12-12 06:11:46 +000032
33namespace android {
34
Mikhail Naganovdfd594e2023-02-08 16:59:41 -080035class StreamOutHalInterfaceCallback;
36class StreamOutHalInterfaceEventCallback;
37class StreamOutHalInterfaceLatencyModeCallback;
38
39// The role of the broker is to connect AIDL callback interface implementations
40// with StreamOut callback implementations. Since AIDL requires all callbacks
41// to be provided upfront, while libaudiohal interfaces allow late registration,
42// there is a need to coordinate the matching process.
43class CallbackBroker : public virtual RefBase {
44 public:
45 virtual ~CallbackBroker() = default;
46 // The cookie is always the stream instance pointer. We don't use weak pointers to avoid extra
47 // costs on reference counting. The stream cleans up related entries on destruction. Since
48 // access to the callbacks map is synchronized, the possibility for pointer aliasing due to
49 // allocation of a new stream at the address of previously deleted stream is avoided.
50 virtual void clearCallbacks(void* cookie) = 0;
51 virtual sp<StreamOutHalInterfaceCallback> getStreamOutCallback(void* cookie) = 0;
52 virtual void setStreamOutCallback(void* cookie, const sp<StreamOutHalInterfaceCallback>&) = 0;
53 virtual sp<StreamOutHalInterfaceEventCallback> getStreamOutEventCallback(void* cookie) = 0;
54 virtual void setStreamOutEventCallback(void* cookie,
55 const sp<StreamOutHalInterfaceEventCallback>&) = 0;
56 virtual sp<StreamOutHalInterfaceLatencyModeCallback> getStreamOutLatencyModeCallback(
57 void* cookie) = 0;
58 virtual void setStreamOutLatencyModeCallback(
59 void* cookie, const sp<StreamOutHalInterfaceLatencyModeCallback>&) = 0;
60};
61
Mikhail Naganovcad0afe2023-03-10 14:25:57 -080062class MicrophoneInfoProvider : public virtual RefBase {
63 public:
64 using Info = std::vector<::aidl::android::media::audio::common::MicrophoneInfo>;
65 virtual ~MicrophoneInfoProvider() = default;
66 // Returns a nullptr if the HAL does not support microphone info retrieval.
67 virtual Info const* getMicrophoneInfo() = 0;
68};
69
Mikhail Naganovdfd594e2023-02-08 16:59:41 -080070class DeviceHalAidl : public DeviceHalInterface, public ConversionHelperAidl,
Mikhail Naganovcad0afe2023-03-10 14:25:57 -080071 public CallbackBroker, public MicrophoneInfoProvider {
Shunkai Yao51202502022-12-12 06:11:46 +000072 public:
Mikhail Naganovf83b9742023-04-24 13:06:04 -070073 status_t getAudioPorts(std::vector<media::audio::common::AudioPort> *ports) override;
74
75 status_t getAudioRoutes(std::vector<media::AudioRoute> *routes) override;
76
Mikhail Naganov1fba38c2023-05-03 17:45:36 -070077 status_t getSupportedModes(std::vector<media::audio::common::AudioMode> *modes) override;
78
Shunkai Yao51202502022-12-12 06:11:46 +000079 // Sets the value of 'devices' to a bitmask of 1 or more values of audio_devices_t.
80 status_t getSupportedDevices(uint32_t *devices) override;
81
82 // Check to see if the audio hardware interface has been initialized.
83 status_t initCheck() override;
84
85 // Set the audio volume of a voice call. Range is between 0.0 and 1.0.
86 status_t setVoiceVolume(float volume) override;
87
88 // Set the audio volume for all audio activities other than voice call.
89 status_t setMasterVolume(float volume) override;
90
91 // Get the current master volume value for the HAL.
92 status_t getMasterVolume(float *volume) override;
93
94 // Called when the audio mode changes.
95 status_t setMode(audio_mode_t mode) override;
96
97 // Muting control.
98 status_t setMicMute(bool state) override;
99
100 status_t getMicMute(bool* state) override;
101
102 status_t setMasterMute(bool state) override;
103
104 status_t getMasterMute(bool *state) override;
105
106 // Set global audio parameters.
107 status_t setParameters(const String8& kvPairs) override;
108
109 // Get global audio parameters.
110 status_t getParameters(const String8& keys, String8 *values) override;
111
112 // Returns audio input buffer size according to parameters passed.
113 status_t getInputBufferSize(const struct audio_config* config, size_t* size) override;
114
115 // Creates and opens the audio hardware output stream. The stream is closed
116 // by releasing all references to the returned object.
117 status_t openOutputStream(audio_io_handle_t handle, audio_devices_t devices,
118 audio_output_flags_t flags, struct audio_config* config,
119 const char* address, sp<StreamOutHalInterface>* outStream) override;
120
121 // Creates and opens the audio hardware input stream. The stream is closed
122 // by releasing all references to the returned object.
123 status_t openInputStream(audio_io_handle_t handle, audio_devices_t devices,
124 struct audio_config* config, audio_input_flags_t flags,
125 const char* address, audio_source_t source,
126 audio_devices_t outputDevice, const char* outputDeviceAddress,
127 sp<StreamInHalInterface>* inStream) override;
128
129 // Returns whether createAudioPatch and releaseAudioPatch operations are supported.
130 status_t supportsAudioPatches(bool* supportsPatches) override;
131
132 // Creates an audio patch between several source and sink ports.
133 status_t createAudioPatch(unsigned int num_sources, const struct audio_port_config* sources,
134 unsigned int num_sinks, const struct audio_port_config* sinks,
135 audio_patch_handle_t* patch) override;
136
137 // Releases an audio patch.
138 status_t releaseAudioPatch(audio_patch_handle_t patch) override;
139
Mikhail Naganov31d46652023-01-10 18:29:25 +0000140 // Fills the list of supported attributes for a given audio port.
141 status_t getAudioPort(struct audio_port* port) override;
142
143 // Fills the list of supported attributes for a given audio port.
144 status_t getAudioPort(struct audio_port_v7 *port) override;
145
Shunkai Yao51202502022-12-12 06:11:46 +0000146 // Set audio port configuration.
147 status_t setAudioPortConfig(const struct audio_port_config* config) override;
148
149 // List microphones
Mikhail Naganove93a0862023-03-15 17:06:59 -0700150 status_t getMicrophones(std::vector<audio_microphone_characteristic_t>* microphones) override;
Shunkai Yao51202502022-12-12 06:11:46 +0000151
Mikhail Naganovd2c7f852023-06-14 18:00:13 -0700152 status_t addDeviceEffect(
153 const struct audio_port_config *device, sp<EffectHalInterface> effect) override;
Shunkai Yao51202502022-12-12 06:11:46 +0000154
Mikhail Naganovd2c7f852023-06-14 18:00:13 -0700155 status_t removeDeviceEffect(
156 const struct audio_port_config *device, sp<EffectHalInterface> effect) override;
Shunkai Yao51202502022-12-12 06:11:46 +0000157
158 status_t getMmapPolicyInfos(media::audio::common::AudioMMapPolicyType policyType __unused,
159 std::vector<media::audio::common::AudioMMapPolicyInfo>* policyInfos
160 __unused) override;
161
162 int32_t getAAudioMixerBurstCount() override;
163
164 int32_t getAAudioHardwareBurstMinUsec() override;
165
166 error::Result<audio_hw_sync_t> getHwAvSync() override;
167
Eric Laurent7af6ee72023-06-29 11:44:54 +0200168 status_t supportsBluetoothVariableLatency(bool* supports __unused) override;
Shunkai Yao51202502022-12-12 06:11:46 +0000169
Vlad Popa03bd5bc2023-01-17 16:16:51 +0100170 status_t getSoundDoseInterface(const std::string& module,
171 ::ndk::SpAIBinder* soundDoseBinder) override;
172
jiabin872de702023-04-27 22:04:31 +0000173 status_t prepareToDisconnectExternalDevice(const struct audio_port_v7 *port) override;
174
Mikhail Naganove93a0862023-03-15 17:06:59 -0700175 status_t setConnectedState(const struct audio_port_v7 *port, bool connected) override;
176
177 status_t setSimulateDeviceConnections(bool enabled) override;
178
jiabin12537fc2023-10-12 17:56:08 +0000179 status_t getAudioMixPort(const struct audio_port_v7* devicePort,
180 struct audio_port_v7* mixPort) override;
181
Mikhail Naganove93a0862023-03-15 17:06:59 -0700182 status_t dump(int __unused, const Vector<String16>& __unused) override;
183
Shunkai Yao51202502022-12-12 06:11:46 +0000184 private:
Mikhail Naganov31d46652023-01-10 18:29:25 +0000185 friend class sp<DeviceHalAidl>;
Mikhail Naganovdfd594e2023-02-08 16:59:41 -0800186
187 struct Callbacks { // No need to use `atomic_wp` because access is serialized.
188 wp<StreamOutHalInterfaceCallback> out;
189 wp<StreamOutHalInterfaceEventCallback> event;
190 wp<StreamOutHalInterfaceLatencyModeCallback> latency;
191 };
Mikhail Naganovcad0afe2023-03-10 14:25:57 -0800192 struct Microphones {
193 enum Status { UNKNOWN, NOT_SUPPORTED, QUERIED };
194 Status status = Status::UNKNOWN;
195 MicrophoneInfoProvider::Info info;
196 };
Mikhail Naganov892f7612023-09-15 18:55:39 -0700197 // IDs of ports for connected external devices, and whether they are held by streams.
198 using ConnectedPorts = std::map<int32_t /*port ID*/, bool>;
Mikhail Naganov5b1eed12023-01-25 11:29:11 -0800199 using Patches = std::map<int32_t /*patch ID*/,
200 ::aidl::android::hardware::audio::core::AudioPatch>;
201 using PortConfigs = std::map<int32_t /*port config ID*/,
202 ::aidl::android::media::audio::common::AudioPortConfig>;
203 using Ports = std::map<int32_t /*port ID*/, ::aidl::android::media::audio::common::AudioPort>;
Mikhail Naganovf83b9742023-04-24 13:06:04 -0700204 using Routes = std::vector<::aidl::android::hardware::audio::core::AudioRoute>;
Mikhail Naganov289468a2023-03-29 10:06:15 -0700205 // Answers the question "whether portID 'first' is reachable from portID 'second'?"
206 // It's not a map because both portIDs are known. The matrix is symmetric.
207 using RoutingMatrix = std::set<std::pair<int32_t, int32_t>>;
Mikhail Naganove93a0862023-03-15 17:06:59 -0700208 using Streams = std::map<wp<StreamHalInterface>, int32_t /*patch ID*/>;
Mikhail Naganov5b1eed12023-01-25 11:29:11 -0800209 class Cleanups;
Mikhail Naganov31d46652023-01-10 18:29:25 +0000210
Mikhail Naganov5b1eed12023-01-25 11:29:11 -0800211 // Must not be constructed directly by clients.
212 DeviceHalAidl(
213 const std::string& instance,
Mikhail Naganove7a26ad2023-05-25 17:36:48 -0700214 const std::shared_ptr<::aidl::android::hardware::audio::core::IModule>& module,
215 const std::shared_ptr<::aidl::android::media::audio::IHalAdapterVendorExtension>& vext);
Shunkai Yao51202502022-12-12 06:11:46 +0000216
Mikhail Naganov31d46652023-01-10 18:29:25 +0000217 ~DeviceHalAidl() override = default;
Mikhail Naganov5b1eed12023-01-25 11:29:11 -0800218
Mikhail Naganov89a9f742023-01-30 12:33:18 -0800219 bool audioDeviceMatches(const ::aidl::android::media::audio::common::AudioDevice& device,
220 const ::aidl::android::media::audio::common::AudioPort& p);
221 bool audioDeviceMatches(const ::aidl::android::media::audio::common::AudioDevice& device,
222 const ::aidl::android::media::audio::common::AudioPortConfig& p);
David Lia8675d42023-03-30 21:08:06 +0800223 status_t createOrUpdatePortConfig(
Mikhail Naganov5b1eed12023-01-25 11:29:11 -0800224 const ::aidl::android::media::audio::common::AudioPortConfig& requestedPortConfig,
David Lia8675d42023-03-30 21:08:06 +0800225 PortConfigs::iterator* result, bool *created);
Mikhail Naganovae9063d2023-11-07 16:43:51 -0800226 void eraseConnectedPort(int32_t portId);
Mikhail Naganove7a26ad2023-05-25 17:36:48 -0700227 status_t filterAndRetrieveBtA2dpParameters(AudioParameter &keys, AudioParameter *result);
Mikhail Naganovccc82112023-04-27 18:14:15 -0700228 status_t filterAndUpdateBtA2dpParameters(AudioParameter &parameters);
229 status_t filterAndUpdateBtHfpParameters(AudioParameter &parameters);
230 status_t filterAndUpdateBtLeParameters(AudioParameter &parameters);
231 status_t filterAndUpdateBtScoParameters(AudioParameter &parameters);
Mikhail Naganove92c34b2023-05-31 14:24:48 -0700232 status_t filterAndUpdateScreenParameters(AudioParameter &parameters);
Mikhail Naganovb9a81312023-07-18 13:55:34 -0700233 status_t filterAndUpdateTelephonyParameters(AudioParameter &parameters);
Mikhail Naganov5b1eed12023-01-25 11:29:11 -0800234 status_t findOrCreatePatch(
235 const std::set<int32_t>& sourcePortConfigIds,
236 const std::set<int32_t>& sinkPortConfigIds,
237 ::aidl::android::hardware::audio::core::AudioPatch* patch, bool* created);
238 status_t findOrCreatePatch(
239 const ::aidl::android::hardware::audio::core::AudioPatch& requestedPatch,
240 ::aidl::android::hardware::audio::core::AudioPatch* patch, bool* created);
241 status_t findOrCreatePortConfig(
242 const ::aidl::android::media::audio::common::AudioDevice& device,
jiabin9c07faf2023-04-26 22:00:44 +0000243 const ::aidl::android::media::audio::common::AudioConfig* config,
Mikhail Naganov5b1eed12023-01-25 11:29:11 -0800244 ::aidl::android::media::audio::common::AudioPortConfig* portConfig,
245 bool* created);
246 status_t findOrCreatePortConfig(
247 const ::aidl::android::media::audio::common::AudioConfig& config,
Mikhail Naganov89a9f742023-01-30 12:33:18 -0800248 const std::optional<::aidl::android::media::audio::common::AudioIoFlags>& flags,
Mikhail Naganov5b1eed12023-01-25 11:29:11 -0800249 int32_t ioHandle,
Mikhail Naganovd8d01f72023-03-09 16:24:40 -0800250 ::aidl::android::media::audio::common::AudioSource aidlSource,
Mikhail Naganov289468a2023-03-29 10:06:15 -0700251 const std::set<int32_t>& destinationPortIds,
Mikhail Naganov5b1eed12023-01-25 11:29:11 -0800252 ::aidl::android::media::audio::common::AudioPortConfig* portConfig, bool* created);
253 status_t findOrCreatePortConfig(
254 const ::aidl::android::media::audio::common::AudioPortConfig& requestedPortConfig,
Mikhail Naganov289468a2023-03-29 10:06:15 -0700255 const std::set<int32_t>& destinationPortIds,
Mikhail Naganov5b1eed12023-01-25 11:29:11 -0800256 ::aidl::android::media::audio::common::AudioPortConfig* portConfig, bool* created);
257 Patches::iterator findPatch(const std::set<int32_t>& sourcePortConfigIds,
258 const std::set<int32_t>& sinkPortConfigIds);
259 Ports::iterator findPort(const ::aidl::android::media::audio::common::AudioDevice& device);
260 Ports::iterator findPort(
261 const ::aidl::android::media::audio::common::AudioConfig& config,
Mikhail Naganov289468a2023-03-29 10:06:15 -0700262 const ::aidl::android::media::audio::common::AudioIoFlags& flags,
263 const std::set<int32_t>& destinationPortIds);
Mikhail Naganov5b1eed12023-01-25 11:29:11 -0800264 PortConfigs::iterator findPortConfig(
265 const ::aidl::android::media::audio::common::AudioDevice& device);
266 PortConfigs::iterator findPortConfig(
jiabin12537fc2023-10-12 17:56:08 +0000267 const std::optional<::aidl::android::media::audio::common::AudioConfig>& config,
Mikhail Naganov89a9f742023-01-30 12:33:18 -0800268 const std::optional<::aidl::android::media::audio::common::AudioIoFlags>& flags,
Mikhail Naganov5b1eed12023-01-25 11:29:11 -0800269 int32_t ioHandle);
Mikhail Naganov66907492023-09-11 17:22:03 -0700270 bool isPortHeldByAStream(int32_t portId);
Mikhail Naganov5b1eed12023-01-25 11:29:11 -0800271 status_t prepareToOpenStream(
272 int32_t aidlHandle,
273 const ::aidl::android::media::audio::common::AudioDevice& aidlDevice,
274 const ::aidl::android::media::audio::common::AudioIoFlags& aidlFlags,
Mikhail Naganovd8d01f72023-03-09 16:24:40 -0800275 ::aidl::android::media::audio::common::AudioSource aidlSource,
Mikhail Naganov5b1eed12023-01-25 11:29:11 -0800276 struct audio_config* config,
277 Cleanups* cleanups,
278 ::aidl::android::media::audio::common::AudioConfig* aidlConfig,
Mikhail Naganov89a9f742023-01-30 12:33:18 -0800279 ::aidl::android::media::audio::common::AudioPortConfig* mixPortConfig,
Mikhail Naganove93a0862023-03-15 17:06:59 -0700280 ::aidl::android::hardware::audio::core::AudioPatch* aidlPatch);
Mikhail Naganov5b1eed12023-01-25 11:29:11 -0800281 void resetPatch(int32_t patchId);
282 void resetPortConfig(int32_t portConfigId);
Mikhail Naganove93a0862023-03-15 17:06:59 -0700283 void resetUnusedPatches();
284 void resetUnusedPatchesAndPortConfigs();
285 void resetUnusedPortConfigs();
Mikhail Naganov289468a2023-03-29 10:06:15 -0700286 status_t updateRoutes();
jiabin12537fc2023-10-12 17:56:08 +0000287 status_t getAudioPort(int32_t portId, ::aidl::android::media::audio::common::AudioPort* port);
Mikhail Naganov5b1eed12023-01-25 11:29:11 -0800288
Mikhail Naganovdfd594e2023-02-08 16:59:41 -0800289 // CallbackBroker implementation
290 void clearCallbacks(void* cookie) override;
291 sp<StreamOutHalInterfaceCallback> getStreamOutCallback(void* cookie) override;
292 void setStreamOutCallback(void* cookie, const sp<StreamOutHalInterfaceCallback>& cb) override;
293 sp<StreamOutHalInterfaceEventCallback> getStreamOutEventCallback(void* cookie) override;
294 void setStreamOutEventCallback(void* cookie,
295 const sp<StreamOutHalInterfaceEventCallback>& cb) override;
296 sp<StreamOutHalInterfaceLatencyModeCallback> getStreamOutLatencyModeCallback(
297 void* cookie) override;
298 void setStreamOutLatencyModeCallback(
299 void* cookie, const sp<StreamOutHalInterfaceLatencyModeCallback>& cb) override;
300 // Implementation helpers.
301 template<class C> sp<C> getCallbackImpl(void* cookie, wp<C> Callbacks::* field);
302 template<class C> void setCallbackImpl(void* cookie, wp<C> Callbacks::* field, const sp<C>& cb);
303
Mikhail Naganovcad0afe2023-03-10 14:25:57 -0800304 // MicrophoneInfoProvider implementation
305 MicrophoneInfoProvider::Info const* getMicrophoneInfo() override;
306
Mikhail Naganov5b1eed12023-01-25 11:29:11 -0800307 const std::string mInstance;
308 const std::shared_ptr<::aidl::android::hardware::audio::core::IModule> mModule;
Mikhail Naganove7a26ad2023-05-25 17:36:48 -0700309 const std::shared_ptr<::aidl::android::media::audio::IHalAdapterVendorExtension> mVendorExt;
Mikhail Naganov1fba38c2023-05-03 17:45:36 -0700310 const std::shared_ptr<::aidl::android::hardware::audio::core::ITelephony> mTelephony;
Mikhail Naganovccc82112023-04-27 18:14:15 -0700311 const std::shared_ptr<::aidl::android::hardware::audio::core::IBluetooth> mBluetooth;
312 const std::shared_ptr<::aidl::android::hardware::audio::core::IBluetoothA2dp> mBluetoothA2dp;
313 const std::shared_ptr<::aidl::android::hardware::audio::core::IBluetoothLe> mBluetoothLe;
Vlad Popa03bd5bc2023-01-17 16:16:51 +0100314 std::shared_ptr<::aidl::android::hardware::audio::core::sounddose::ISoundDose>
315 mSoundDose = nullptr;
Mikhail Naganov5b1eed12023-01-25 11:29:11 -0800316 Ports mPorts;
Mikhail Naganovae9063d2023-11-07 16:43:51 -0800317 // Remote submix "template" ports (no address specified, no profiles).
318 // They are excluded from `mPorts` as their presence confuses the framework code.
319 std::optional<::aidl::android::media::audio::common::AudioPort> mRemoteSubmixIn;
320 std::optional<::aidl::android::media::audio::common::AudioPort> mRemoteSubmixOut;
Mikhail Naganov81bf4d02023-02-06 12:20:38 -0800321 int32_t mDefaultInputPortId = -1;
322 int32_t mDefaultOutputPortId = -1;
Mikhail Naganov5b1eed12023-01-25 11:29:11 -0800323 PortConfigs mPortConfigs;
jiabin9c07faf2023-04-26 22:00:44 +0000324 std::set<int32_t> mInitialPortConfigIds;
Mikhail Naganov5b1eed12023-01-25 11:29:11 -0800325 Patches mPatches;
Mikhail Naganovf83b9742023-04-24 13:06:04 -0700326 Routes mRoutes;
Mikhail Naganov289468a2023-03-29 10:06:15 -0700327 RoutingMatrix mRoutingMatrix;
Mikhail Naganove93a0862023-03-15 17:06:59 -0700328 Streams mStreams;
Mikhail Naganovcad0afe2023-03-10 14:25:57 -0800329 Microphones mMicrophones;
Mikhail Naganovdfd594e2023-02-08 16:59:41 -0800330 std::mutex mLock;
331 std::map<void*, Callbacks> mCallbacks GUARDED_BY(mLock);
jiabin872de702023-04-27 22:04:31 +0000332 std::set<audio_port_handle_t> mDeviceDisconnectionNotified;
Mikhail Naganov892f7612023-09-15 18:55:39 -0700333 ConnectedPorts mConnectedPorts;
Mikhail Naganovae9063d2023-11-07 16:43:51 -0800334 std::pair<int32_t, ::aidl::android::media::audio::common::AudioPort>
335 mDisconnectedPortReplacement;
Shunkai Yao51202502022-12-12 06:11:46 +0000336};
337
338} // namespace android