Loading Android.bp +1 −0 Original line number Diff line number Diff line Loading @@ -97,6 +97,7 @@ filegroup { // AIDL sources from external directories ":android.hardware.biometrics.common-V4-java-source", ":android.hardware.biometrics.fingerprint-V3-java-source", ":android.hardware.biometrics.face-V4-java-source", ":android.hardware.gnss-V2-java-source", ":android.hardware.graphics.common-V3-java-source", ":android.hardware.keymaster-V4-java-source", Loading core/java/android/hardware/face/FaceSensorConfigurations.aidl 0 → 100644 +18 −0 Original line number Diff line number Diff line /* * Copyright (C) 2023 The Android Open Source Project * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. * You may obtain a copy of the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. * See the License for the specific language governing permissions and * limitations under the License. */ package android.hardware.face; parcelable FaceSensorConfigurations; No newline at end of file core/java/android/hardware/face/FaceSensorConfigurations.java 0 → 100644 +182 −0 Original line number Diff line number Diff line /* * Copyright (C) 2023 The Android Open Source Project * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. * You may obtain a copy of the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. * See the License for the specific language governing permissions and * limitations under the License. */ package android.hardware.face; import static android.hardware.biometrics.BiometricAuthenticator.TYPE_FACE; import android.annotation.Nullable; import android.content.Context; import android.hardware.biometrics.face.IFace; import android.hardware.biometrics.face.SensorProps; import android.os.Parcel; import android.os.Parcelable; import android.os.RemoteException; import android.util.Log; import android.util.Pair; import android.util.Slog; import androidx.annotation.NonNull; import java.util.ArrayList; import java.util.HashMap; import java.util.List; import java.util.Map; import java.util.Optional; import java.util.function.Function; /** * Provides the sensor props for face sensor, if available. * @hide */ public class FaceSensorConfigurations implements Parcelable { private static final String TAG = "FaceSensorConfigurations"; private final boolean mResetLockoutRequiresChallenge; private final Map<String, SensorProps[]> mSensorPropsMap; public static final Creator<FaceSensorConfigurations> CREATOR = new Creator<FaceSensorConfigurations>() { @Override public FaceSensorConfigurations createFromParcel(Parcel in) { return new FaceSensorConfigurations(in); } @Override public FaceSensorConfigurations[] newArray(int size) { return new FaceSensorConfigurations[size]; } }; public FaceSensorConfigurations(boolean resetLockoutRequiresChallenge) { mResetLockoutRequiresChallenge = resetLockoutRequiresChallenge; mSensorPropsMap = new HashMap<>(); } protected FaceSensorConfigurations(Parcel in) { mResetLockoutRequiresChallenge = in.readByte() != 0; mSensorPropsMap = in.readHashMap(null, String.class, SensorProps[].class); } /** * Process AIDL instances to extract sensor props and add it to the sensor map. * @param aidlInstances available face AIDL instances * @param getIFace function that provides the daemon for the specific instance */ public void addAidlConfigs(@NonNull String[] aidlInstances, @NonNull Function<String, IFace> getIFace) { for (String aidlInstance : aidlInstances) { final String fqName = IFace.DESCRIPTOR + "/" + aidlInstance; IFace face = getIFace.apply(fqName); try { if (face != null) { mSensorPropsMap.put(aidlInstance, face.getSensorProps()); } else { Slog.e(TAG, "Unable to get declared service: " + fqName); } } catch (RemoteException e) { Log.d(TAG, "Unable to get sensor properties!"); } } } /** * Parse through HIDL configuration and add it to the sensor map. */ public void addHidlConfigs(@NonNull String[] hidlConfigStrings, @NonNull Context context) { final List<HidlFaceSensorConfig> hidlFaceSensorConfigs = new ArrayList<>(); for (String hidlConfig: hidlConfigStrings) { final HidlFaceSensorConfig hidlFaceSensorConfig = new HidlFaceSensorConfig(); try { hidlFaceSensorConfig.parse(hidlConfig, context); } catch (Exception e) { Log.e(TAG, "HIDL sensor configuration format is incorrect."); continue; } if (hidlFaceSensorConfig.getModality() == TYPE_FACE) { hidlFaceSensorConfigs.add(hidlFaceSensorConfig); } } final String hidlHalInstanceName = "defaultHIDL"; mSensorPropsMap.put(hidlHalInstanceName, hidlFaceSensorConfigs.toArray( new SensorProps[hidlFaceSensorConfigs.size()])); } /** * Returns true if any face sensors have been added. */ public boolean hasSensorConfigurations() { return mSensorPropsMap.size() > 0; } /** * Returns true if there is only a single face sensor configuration available. */ public boolean isSingleSensorConfigurationPresent() { return mSensorPropsMap.size() == 1; } /** * Return sensor props for the given instance. If instance is not available, * then null is returned. */ @Nullable public Pair<String, SensorProps[]> getSensorPairForInstance(String instance) { if (mSensorPropsMap.containsKey(instance)) { return new Pair<>(instance, mSensorPropsMap.get(instance)); } return null; } /** * Return the first pair of instance and sensor props, which does not correspond to the given * If instance is not available, then null is returned. */ @Nullable public Pair<String, SensorProps[]> getSensorPairNotForInstance(String instance) { Optional<String> notAVirtualInstance = mSensorPropsMap.keySet().stream().filter( (instanceName) -> !instanceName.equals(instance)).findFirst(); return notAVirtualInstance.map(this::getSensorPairForInstance).orElseGet( this::getSensorPair); } /** * Returns the first pair of instance and sensor props that has been added to the map. */ @Nullable public Pair<String, SensorProps[]> getSensorPair() { Optional<String> optionalInstance = mSensorPropsMap.keySet().stream().findFirst(); return optionalInstance.map(this::getSensorPairForInstance).orElse(null); } public boolean getResetLockoutRequiresChallenge() { return mResetLockoutRequiresChallenge; } @Override public int describeContents() { return 0; } @Override public void writeToParcel(@NonNull Parcel dest, int flags) { dest.writeByte((byte) (mResetLockoutRequiresChallenge ? 1 : 0)); dest.writeMap(mSensorPropsMap); } } core/java/android/hardware/face/HidlFaceSensorConfig.java 0 → 100644 +85 −0 Original line number Diff line number Diff line /* * Copyright (C) 2023 The Android Open Source Project * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. * You may obtain a copy of the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. * See the License for the specific language governing permissions and * limitations under the License. */ package android.hardware.face; import android.annotation.NonNull; import android.content.Context; import android.hardware.biometrics.BiometricAuthenticator; import android.hardware.biometrics.BiometricManager; import android.hardware.biometrics.common.CommonProps; import android.hardware.biometrics.common.SensorStrength; import android.hardware.biometrics.face.SensorProps; import com.android.internal.R; /** * Parse HIDL face sensor config and map it to SensorProps.aidl to match AIDL. * See core/res/res/values/config.xml config_biometric_sensors * @hide */ public final class HidlFaceSensorConfig extends SensorProps { private int mSensorId; private int mModality; private int mStrength; /** * Parse through the config string and map it to SensorProps.aidl. * @throws IllegalArgumentException when config string has unexpected format */ public void parse(@NonNull String config, @NonNull Context context) throws IllegalArgumentException { final String[] elems = config.split(":"); if (elems.length < 3) { throw new IllegalArgumentException(); } mSensorId = Integer.parseInt(elems[0]); mModality = Integer.parseInt(elems[1]); mStrength = Integer.parseInt(elems[2]); mapHidlToAidlFaceSensorConfigurations(context); } @BiometricAuthenticator.Modality public int getModality() { return mModality; } private void mapHidlToAidlFaceSensorConfigurations(@NonNull Context context) { commonProps = new CommonProps(); commonProps.sensorId = mSensorId; commonProps.sensorStrength = authenticatorStrengthToPropertyStrength(mStrength); halControlsPreview = context.getResources().getBoolean( R.bool.config_faceAuthSupportsSelfIllumination); commonProps.maxEnrollmentsPerUser = context.getResources().getInteger( R.integer.config_faceMaxTemplatesPerUser); commonProps.componentInfo = null; supportsDetectInteraction = false; } private byte authenticatorStrengthToPropertyStrength( @BiometricManager.Authenticators.Types int strength) { switch (strength) { case BiometricManager.Authenticators.BIOMETRIC_CONVENIENCE: return SensorStrength.CONVENIENCE; case BiometricManager.Authenticators.BIOMETRIC_WEAK: return SensorStrength.WEAK; case BiometricManager.Authenticators.BIOMETRIC_STRONG: return SensorStrength.STRONG; default: throw new IllegalArgumentException("Unknown strength: " + strength); } } } core/java/android/hardware/face/IFaceService.aidl +5 −0 Original line number Diff line number Diff line Loading @@ -26,6 +26,7 @@ import android.hardware.face.IFaceServiceReceiver; import android.hardware.face.Face; import android.hardware.face.FaceAuthenticateOptions; import android.hardware.face.FaceSensorPropertiesInternal; import android.hardware.face.FaceSensorConfigurations; import android.view.Surface; /** Loading Loading @@ -167,6 +168,10 @@ interface IFaceService { @EnforcePermission("USE_BIOMETRIC_INTERNAL") void registerAuthenticators(in List<FaceSensorPropertiesInternal> hidlSensors); //Register all available face sensors. @EnforcePermission("USE_BIOMETRIC_INTERNAL") void registerAuthenticatorsLegacy(in FaceSensorConfigurations faceSensorConfigurations); // Adds a callback which gets called when the service registers all of the face // authenticators. The callback is automatically removed after it's invoked. void addAuthenticatorsRegisteredCallback(IFaceAuthenticatorsRegisteredCallback callback); Loading Loading
Android.bp +1 −0 Original line number Diff line number Diff line Loading @@ -97,6 +97,7 @@ filegroup { // AIDL sources from external directories ":android.hardware.biometrics.common-V4-java-source", ":android.hardware.biometrics.fingerprint-V3-java-source", ":android.hardware.biometrics.face-V4-java-source", ":android.hardware.gnss-V2-java-source", ":android.hardware.graphics.common-V3-java-source", ":android.hardware.keymaster-V4-java-source", Loading
core/java/android/hardware/face/FaceSensorConfigurations.aidl 0 → 100644 +18 −0 Original line number Diff line number Diff line /* * Copyright (C) 2023 The Android Open Source Project * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. * You may obtain a copy of the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. * See the License for the specific language governing permissions and * limitations under the License. */ package android.hardware.face; parcelable FaceSensorConfigurations; No newline at end of file
core/java/android/hardware/face/FaceSensorConfigurations.java 0 → 100644 +182 −0 Original line number Diff line number Diff line /* * Copyright (C) 2023 The Android Open Source Project * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. * You may obtain a copy of the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. * See the License for the specific language governing permissions and * limitations under the License. */ package android.hardware.face; import static android.hardware.biometrics.BiometricAuthenticator.TYPE_FACE; import android.annotation.Nullable; import android.content.Context; import android.hardware.biometrics.face.IFace; import android.hardware.biometrics.face.SensorProps; import android.os.Parcel; import android.os.Parcelable; import android.os.RemoteException; import android.util.Log; import android.util.Pair; import android.util.Slog; import androidx.annotation.NonNull; import java.util.ArrayList; import java.util.HashMap; import java.util.List; import java.util.Map; import java.util.Optional; import java.util.function.Function; /** * Provides the sensor props for face sensor, if available. * @hide */ public class FaceSensorConfigurations implements Parcelable { private static final String TAG = "FaceSensorConfigurations"; private final boolean mResetLockoutRequiresChallenge; private final Map<String, SensorProps[]> mSensorPropsMap; public static final Creator<FaceSensorConfigurations> CREATOR = new Creator<FaceSensorConfigurations>() { @Override public FaceSensorConfigurations createFromParcel(Parcel in) { return new FaceSensorConfigurations(in); } @Override public FaceSensorConfigurations[] newArray(int size) { return new FaceSensorConfigurations[size]; } }; public FaceSensorConfigurations(boolean resetLockoutRequiresChallenge) { mResetLockoutRequiresChallenge = resetLockoutRequiresChallenge; mSensorPropsMap = new HashMap<>(); } protected FaceSensorConfigurations(Parcel in) { mResetLockoutRequiresChallenge = in.readByte() != 0; mSensorPropsMap = in.readHashMap(null, String.class, SensorProps[].class); } /** * Process AIDL instances to extract sensor props and add it to the sensor map. * @param aidlInstances available face AIDL instances * @param getIFace function that provides the daemon for the specific instance */ public void addAidlConfigs(@NonNull String[] aidlInstances, @NonNull Function<String, IFace> getIFace) { for (String aidlInstance : aidlInstances) { final String fqName = IFace.DESCRIPTOR + "/" + aidlInstance; IFace face = getIFace.apply(fqName); try { if (face != null) { mSensorPropsMap.put(aidlInstance, face.getSensorProps()); } else { Slog.e(TAG, "Unable to get declared service: " + fqName); } } catch (RemoteException e) { Log.d(TAG, "Unable to get sensor properties!"); } } } /** * Parse through HIDL configuration and add it to the sensor map. */ public void addHidlConfigs(@NonNull String[] hidlConfigStrings, @NonNull Context context) { final List<HidlFaceSensorConfig> hidlFaceSensorConfigs = new ArrayList<>(); for (String hidlConfig: hidlConfigStrings) { final HidlFaceSensorConfig hidlFaceSensorConfig = new HidlFaceSensorConfig(); try { hidlFaceSensorConfig.parse(hidlConfig, context); } catch (Exception e) { Log.e(TAG, "HIDL sensor configuration format is incorrect."); continue; } if (hidlFaceSensorConfig.getModality() == TYPE_FACE) { hidlFaceSensorConfigs.add(hidlFaceSensorConfig); } } final String hidlHalInstanceName = "defaultHIDL"; mSensorPropsMap.put(hidlHalInstanceName, hidlFaceSensorConfigs.toArray( new SensorProps[hidlFaceSensorConfigs.size()])); } /** * Returns true if any face sensors have been added. */ public boolean hasSensorConfigurations() { return mSensorPropsMap.size() > 0; } /** * Returns true if there is only a single face sensor configuration available. */ public boolean isSingleSensorConfigurationPresent() { return mSensorPropsMap.size() == 1; } /** * Return sensor props for the given instance. If instance is not available, * then null is returned. */ @Nullable public Pair<String, SensorProps[]> getSensorPairForInstance(String instance) { if (mSensorPropsMap.containsKey(instance)) { return new Pair<>(instance, mSensorPropsMap.get(instance)); } return null; } /** * Return the first pair of instance and sensor props, which does not correspond to the given * If instance is not available, then null is returned. */ @Nullable public Pair<String, SensorProps[]> getSensorPairNotForInstance(String instance) { Optional<String> notAVirtualInstance = mSensorPropsMap.keySet().stream().filter( (instanceName) -> !instanceName.equals(instance)).findFirst(); return notAVirtualInstance.map(this::getSensorPairForInstance).orElseGet( this::getSensorPair); } /** * Returns the first pair of instance and sensor props that has been added to the map. */ @Nullable public Pair<String, SensorProps[]> getSensorPair() { Optional<String> optionalInstance = mSensorPropsMap.keySet().stream().findFirst(); return optionalInstance.map(this::getSensorPairForInstance).orElse(null); } public boolean getResetLockoutRequiresChallenge() { return mResetLockoutRequiresChallenge; } @Override public int describeContents() { return 0; } @Override public void writeToParcel(@NonNull Parcel dest, int flags) { dest.writeByte((byte) (mResetLockoutRequiresChallenge ? 1 : 0)); dest.writeMap(mSensorPropsMap); } }
core/java/android/hardware/face/HidlFaceSensorConfig.java 0 → 100644 +85 −0 Original line number Diff line number Diff line /* * Copyright (C) 2023 The Android Open Source Project * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. * You may obtain a copy of the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. * See the License for the specific language governing permissions and * limitations under the License. */ package android.hardware.face; import android.annotation.NonNull; import android.content.Context; import android.hardware.biometrics.BiometricAuthenticator; import android.hardware.biometrics.BiometricManager; import android.hardware.biometrics.common.CommonProps; import android.hardware.biometrics.common.SensorStrength; import android.hardware.biometrics.face.SensorProps; import com.android.internal.R; /** * Parse HIDL face sensor config and map it to SensorProps.aidl to match AIDL. * See core/res/res/values/config.xml config_biometric_sensors * @hide */ public final class HidlFaceSensorConfig extends SensorProps { private int mSensorId; private int mModality; private int mStrength; /** * Parse through the config string and map it to SensorProps.aidl. * @throws IllegalArgumentException when config string has unexpected format */ public void parse(@NonNull String config, @NonNull Context context) throws IllegalArgumentException { final String[] elems = config.split(":"); if (elems.length < 3) { throw new IllegalArgumentException(); } mSensorId = Integer.parseInt(elems[0]); mModality = Integer.parseInt(elems[1]); mStrength = Integer.parseInt(elems[2]); mapHidlToAidlFaceSensorConfigurations(context); } @BiometricAuthenticator.Modality public int getModality() { return mModality; } private void mapHidlToAidlFaceSensorConfigurations(@NonNull Context context) { commonProps = new CommonProps(); commonProps.sensorId = mSensorId; commonProps.sensorStrength = authenticatorStrengthToPropertyStrength(mStrength); halControlsPreview = context.getResources().getBoolean( R.bool.config_faceAuthSupportsSelfIllumination); commonProps.maxEnrollmentsPerUser = context.getResources().getInteger( R.integer.config_faceMaxTemplatesPerUser); commonProps.componentInfo = null; supportsDetectInteraction = false; } private byte authenticatorStrengthToPropertyStrength( @BiometricManager.Authenticators.Types int strength) { switch (strength) { case BiometricManager.Authenticators.BIOMETRIC_CONVENIENCE: return SensorStrength.CONVENIENCE; case BiometricManager.Authenticators.BIOMETRIC_WEAK: return SensorStrength.WEAK; case BiometricManager.Authenticators.BIOMETRIC_STRONG: return SensorStrength.STRONG; default: throw new IllegalArgumentException("Unknown strength: " + strength); } } }
core/java/android/hardware/face/IFaceService.aidl +5 −0 Original line number Diff line number Diff line Loading @@ -26,6 +26,7 @@ import android.hardware.face.IFaceServiceReceiver; import android.hardware.face.Face; import android.hardware.face.FaceAuthenticateOptions; import android.hardware.face.FaceSensorPropertiesInternal; import android.hardware.face.FaceSensorConfigurations; import android.view.Surface; /** Loading Loading @@ -167,6 +168,10 @@ interface IFaceService { @EnforcePermission("USE_BIOMETRIC_INTERNAL") void registerAuthenticators(in List<FaceSensorPropertiesInternal> hidlSensors); //Register all available face sensors. @EnforcePermission("USE_BIOMETRIC_INTERNAL") void registerAuthenticatorsLegacy(in FaceSensorConfigurations faceSensorConfigurations); // Adds a callback which gets called when the service registers all of the face // authenticators. The callback is automatically removed after it's invoked. void addAuthenticatorsRegisteredCallback(IFaceAuthenticatorsRegisteredCallback callback); Loading