/* * Copyright (C) 2008 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 com.android.provision; import static android.app.admin.DevicePolicyManager.ACTION_PROVISION_MANAGED_DEVICE_FROM_TRUSTED_SOURCE; import static android.app.admin.DevicePolicyManager.EXTRA_PROVISIONING_DEVICE_ADMIN_COMPONENT_NAME; import static android.app.admin.DevicePolicyManager.EXTRA_PROVISIONING_DEVICE_ADMIN_PACKAGE_DOWNLOAD_LOCATION; import static android.app.admin.DevicePolicyManager.EXTRA_PROVISIONING_DEVICE_ADMIN_SIGNATURE_CHECKSUM; import static android.app.admin.DevicePolicyManager.EXTRA_PROVISIONING_TRIGGER; import static com.android.provision.Utils.DEFAULT_SETTINGS_PROVISION_DO_MODE; import static com.android.provision.Utils.SETTINGS_PROVISION_DO_MODE; import static com.android.provision.Utils.TAG; import static com.android.provision.Utils.getSettings; import android.app.Activity; import android.app.AlertDialog; import android.app.admin.DevicePolicyManager; import android.content.ComponentName; import android.content.Intent; import android.content.pm.PackageManager; import android.os.Bundle; import android.provider.Settings; import android.util.Log; /** * Application that sets the provisioned bit, like {@code SetupWizard} does. * *
By default, it silently provisions the device, but it can also be used to provision * {@code DeviceOwner}. For example, to set the {@code TestDPC} app, run the steps below: *
adb root
adb install PATH_TO_TESTDPC_APK
adb shell settings put secure tmp_provision_set_do 1
adb shell settings put secure tmp_provision_package com.afwsamples.testdpc
adb shell settings put secure tmp_provision_receiver com.afwsamples.testdpc.DeviceAdminReceiver
adb shell settings put secure tmp_provision_trigger 2
adb shell rm /data/system/device_policies.xml
adb shell settings put global device_provisioned 0
adb shell settings put secure user_setup_complete 0
adb shell pm enable com.android.provision
adb shell pm enable com.android.provision/.DefaultActivity
adb shell stop && adb shell start
// You might also need to run:
adb shell am start com.android.provision/.DefaultActivity
*
*/
public class DefaultActivity extends Activity {
// TODO(b/170333009): copied from ManagedProvisioning app, as they're hidden;
private static final String PROVISION_FINALIZATION_INSIDE_SUW =
"android.app.action.PROVISION_FINALIZATION_INSIDE_SUW";
private static final int RESULT_CODE_PROFILE_OWNER_SET = 122;
private static final int RESULT_CODE_DEVICE_OWNER_SET = 123;
private static final int REQUEST_CODE_STEP1 = 42;
private static final int REQUEST_CODE_STEP2_PO = 43;
private static final int REQUEST_CODE_STEP2_DO = 44;
@Override
protected void onCreate(Bundle icicle) {
super.onCreate(icicle);
boolean provisionDeviceOwner = getSettings(getContentResolver(), SETTINGS_PROVISION_DO_MODE,
DEFAULT_SETTINGS_PROVISION_DO_MODE) == 1;
if (provisionDeviceOwner) {
provisionDeviceOwner();
return;
}
finishSetup();
}
private void finishSetup() {
setProvisioningState();
disableSelfAndFinish();
}
private void setProvisioningState() {
Log.i(TAG, "Setting provisioning state");
// Add a persistent setting to allow other apps to know the device has been provisioned.
Settings.Global.putInt(getContentResolver(), Settings.Global.DEVICE_PROVISIONED, 1);
Settings.Secure.putInt(getContentResolver(), Settings.Secure.USER_SETUP_COMPLETE, 1);
}
private void disableSelfAndFinish() {
// remove this activity from the package manager.
PackageManager pm = getPackageManager();
ComponentName name = new ComponentName(this, DefaultActivity.class);
Log.i(TAG, "Disabling itself (" + name + ")");
pm.setComponentEnabledSetting(name, PackageManager.COMPONENT_ENABLED_STATE_DISABLED,
PackageManager.DONT_KILL_APP);
// terminate the activity.
finish();
}
private void provisionDeviceOwner() {
if (!getPackageManager()
.hasSystemFeature(PackageManager.FEATURE_DEVICE_ADMIN)) {
Log.e(TAG, "Cannot set up device owner because device does not have the "
+ PackageManager.FEATURE_DEVICE_ADMIN + " feature");
finishSetup();
return;
}
DevicePolicyManager dpm = getSystemService(DevicePolicyManager.class);
if (!dpm.isProvisioningAllowed(DevicePolicyManager.ACTION_PROVISION_MANAGED_DEVICE)) {
Log.e(TAG, "DeviceOwner provisioning is not allowed, most like device is already "
+ "provisioned");
finishSetup();
return;
}
DpcInfo dpcInfo = new DpcInfo(getContentResolver());
Intent intent = new Intent(ACTION_PROVISION_MANAGED_DEVICE_FROM_TRUSTED_SOURCE);
intent.putExtra(EXTRA_PROVISIONING_TRIGGER, dpcInfo.trigger);
intent.putExtra(EXTRA_PROVISIONING_DEVICE_ADMIN_COMPONENT_NAME,
dpcInfo.getReceiverComponentName());
if (dpcInfo.checkSum != null) {
intent.putExtra(EXTRA_PROVISIONING_DEVICE_ADMIN_SIGNATURE_CHECKSUM, dpcInfo.checkSum);
}
if (dpcInfo.downloadUrl != null) {
intent.putExtra(EXTRA_PROVISIONING_DEVICE_ADMIN_PACKAGE_DOWNLOAD_LOCATION,
dpcInfo.downloadUrl);
}
Log.i(TAG, "Provisioning device with " + dpcInfo + ". Intent: " + intent);
startActivityForResult(intent, REQUEST_CODE_STEP1);
}
@Override
protected void onActivityResult(int requestCode, int resultCode, Intent data) {
Log.d(TAG, "onActivityResult(): request=" + requestCode + ", result="
+ resultCodeToString(resultCode) + ", data=" + data);
switch (requestCode) {
case REQUEST_CODE_STEP1:
onProvisioningStep1Result(resultCode);
break;
case REQUEST_CODE_STEP2_PO:
case REQUEST_CODE_STEP2_DO:
onProvisioningStep2Result(requestCode, resultCode);
break;
default:
showErrorMessage("onActivityResult(): invalid request code " + requestCode);
}
}
private void onProvisioningStep1Result(int resultCode) {
int requestCodeStep2;
switch (resultCode) {
case RESULT_CODE_PROFILE_OWNER_SET:
requestCodeStep2 = REQUEST_CODE_STEP2_PO;
break;
case RESULT_CODE_DEVICE_OWNER_SET:
requestCodeStep2 = REQUEST_CODE_STEP2_DO;
break;
default:
factoryReset("invalid response from "
+ ACTION_PROVISION_MANAGED_DEVICE_FROM_TRUSTED_SOURCE + ": "
+ resultCodeToString(resultCode));
return;
}
Intent intent = new Intent(PROVISION_FINALIZATION_INSIDE_SUW)
.addCategory(Intent.CATEGORY_DEFAULT);
Log.i(TAG, "Finalizing DPC with " + intent);
startActivityForResult(intent, requestCodeStep2);
}
private void onProvisioningStep2Result(int requestCode, int resultCode) {
// Must set state before launching the intent that finalize the DPC, because the DPC
// implementation might not remove the back button
setProvisioningState();
boolean doMode = requestCode == REQUEST_CODE_STEP2_DO;
if (resultCode != RESULT_OK) {
factoryReset("invalid response from " + PROVISION_FINALIZATION_INSIDE_SUW + ": "
+ resultCodeToString(resultCode));
return;
}
Log.i(TAG, (doMode ? "Device owner" : "Profile owner") + " mode provisioned!");
disableSelfAndFinish();
}
private static String resultCodeToString(int resultCode) {
StringBuilder result = new StringBuilder();
switch (resultCode) {
case RESULT_OK:
result.append("RESULT_OK");
break;
case RESULT_CANCELED:
result.append("RESULT_CANCELED");
break;
case RESULT_FIRST_USER:
result.append("RESULT_FIRST_USER");
break;
case RESULT_CODE_PROFILE_OWNER_SET:
result.append("RESULT_CODE_PROFILE_OWNER_SET");
break;
case RESULT_CODE_DEVICE_OWNER_SET:
result.append("RESULT_CODE_DEVICE_OWNER_SET");
break;
default:
result.append("UNKNOWN_CODE");
}
return result.append('(').append(resultCode).append(')').toString();
}
private void showErrorMessage(String message) {
Log.e(TAG, "Error: " + message);
}
private void factoryReset(String reason) {
new AlertDialog.Builder(this)
.setMessage("Device owner provisioning failed (" + reason
+ ") and device must be factory reset")
.setPositiveButton("Reset", (d, w) -> sendFactoryResetIntent(reason))
.setOnDismissListener((d) -> sendFactoryResetIntent(reason))
.show();
}
private void sendFactoryResetIntent(String reason) {
Log.e(TAG, "Factory resetting: " + reason);
Intent intent = new Intent(Intent.ACTION_FACTORY_RESET);
intent.setPackage("android");
intent.addFlags(Intent.FLAG_RECEIVER_FOREGROUND);
intent.putExtra(Intent.EXTRA_REASON, reason);
sendBroadcast(intent);
// Just in case the factory reset request fails...
finishSetup();
}
}