We usually get the SystemService with the following code

        ActivityManager activityManager= (ActivityManager) getSystemService(Context.ACTIVITY_SERVICE);
        activityManager.getRunningAppProcesses();
Copy the code

And related implementation is in ContextImpl getSystemService: frameworks/base/core/Java/android/app/ContextImpl. Java, HTML code is as follows:

@Override public Object getSystemService(String name) { if (vmIncorrectContextUseEnabled()) { // Check incorrect Context  usage. if (isUiComponent(name) && ! isSelfOrOuterUiContext()) { final String errorMessage = "Tried to access visual service " + SystemServiceRegistry.getSystemServiceClassName(name) + " from a non-visual Context:" + getOuterContext(); final String message = "Visual services, such as WindowManager, WallpaperService " + "or LayoutInflater should be accessed from Activity or other visual " + "Context. Use an Activity or a Context created with " + "Context#createWindowContext(int, Bundle), which are adjusted to " + "the configuration and visual bounds of an area on screen."; final Exception exception = new IllegalAccessException(errorMessage); StrictMode.onIncorrectContextUsed(message, exception); Log.e(TAG, errorMessage + " " + message, exception); } } return SystemServiceRegistry.getSystemService(this, name); }Copy the code

Can see is ultimately through SystemServiceRegistry (frameworks/base/core/Java/android/app/SystemServiceRegistry. Java) classes to obtain, From the name of the class, we can infer that there must be a related subscription method in the class. After checking, we can find that there is a subscription function:

    private static <T> void registerService(@NonNull String serviceName,
            @NonNull Class<T> serviceClass, @NonNull ServiceFetcher<T> serviceFetcher) {
        SYSTEM_SERVICE_NAMES.put(serviceClass, serviceName);
        SYSTEM_SERVICE_FETCHERS.put(serviceName, serviceFetcher);
        SYSTEM_SERVICE_CLASS_NAMES.put(serviceName, serviceClass.getSimpleName());
    }
Copy the code

And in this class, we search the corresponding ActivityManager and find that many managers are registered by calling registerService in the static code block within the class:

static { ...... Omit... registerService(Context.ACTIVITY_SERVICE, ActivityManager.class, new CachedServiceFetcher<ActivityManager>() { @Override public ActivityManager createService(ContextImpl ctx) { return new ActivityManager(ctx.getOuterContext(), ctx.mMainThread.getHandler()); }}); }... Omit...Copy the code

ActivityManager is only a proxy class. The Binder implementation Service provides the implementation of the call. Next, we implement a SystemService by ourselves, and provide the corresponding Java package for the application to call.

Write AIDL files

We imitate ActivityManager also will create the AIDL frameworks/base/core/Java/android/app/directory frameworks/base/core/java/android/app/yy/ITestService.aidl:

package android.app.yy;
/**
 * @hide
 * */
interface ITestService {
    String todo();
}
Copy the code

Add the servicename

Then we’ll supplement the Context with static variables that we use to get Manager, like context.activity_service, which we pass in getSystemService and registerService as keys, To do a mapping lookup. Frameworks/base/core/Java/android/content/Context. Java:

public static final String TEST_SERVICE = "test"; /** * Contex also has a custom @servicename annotation, which is used to restrict the context.getSystemService argument: *public abstract @Nullable Object getSystemService(@ServiceName @NonNull String name); * So for consistency and standardization, we also add Service to @servicename's definition: **/ /** @hide */ @StringDef(suffix = { "_SERVICE" }, value = { //...... Omit... TEST_SERVICE, ACTIVITY_SERVICE, //...... Omit... }) @Retention(RetentionPolicy.SOURCE) public @interface ServiceName {}Copy the code

Implementing Service logic

Since AIDL is already written, Binder proxy classes are generated when compiled, we can integrate the Stub to write the Service logic: To frameworks/base/services/core/Java/com/android/server/directory, you can see a am directory, ams is implemented under the path, we imitate the create our service at the path: frameworks/base/services/core/java/com/android/server/yy/TestService.java

package com.android.server.yy; import android.content.Context; import android.app.yy.ITestService; public class TestService extends ITestService.Stub { private final Context mContext; public TestService(Context context) { super(); mContext = context; } @override public String todo() {return "todo succeeded "; }}Copy the code

Registration Service

All Services (Binder implementations) are created in the SystemServer process and registered with the ServiceManager process. All the Binder in the Android interactions are driven by Binder to find corresponding ServiceManager Service communication, frameworks/base/services/Java/com/Android/server/SystemSer ver.java:

/** * SystemServer run encapsulates three functions for starting different types of services. ** AMS is created and registered in startBootstrapServices. * Add our custom Service to startOtherServices: */ private void run() { // Start services. try { t.traceBegin("StartServices"); startBootstrapServices(t); startCoreServices(t); startOtherServices(t); } catch (Throwable ex) { Slog.e("System", "******************************************"); Slog.e("System", "************ Failure starting system services", ex); throw ex; } finally { t.traceEnd(); // StartServices } } private void startOtherServices(@NonNull TimingsTraceAndSlog t) { //...... Omit... TestService testService=new TestService(context); ServiceManager.addService(Context.TEST_SERVICE,testService); / /... Omit... }Copy the code

Implementation Manager

Next, we can realize the corresponding Manager, in order to provide for the use of application side, Manager in the same path and AIDL, frameworks/base/core/Java/android/app/yy/TestServiceManager Java

package android.app.yy; import android.os.IBinder; import android.os.ServiceManager; import android.app.yy.ITestService; import android.content.Context; import android.os.RemoteException; import android.compat.annotation.UnsupportedAppUsage; import android.annotation.Nullable; import android.os.ServiceManager.ServiceNotFoundException; public class TestServiceManager { private static TestServiceManager instance; private final ITestService mService; private Context mContext; /** * @hide */ private TestServiceManager(ITestService service,Context context) { mService = service; mContext=context; } /** * @hide */ @UnsupportedAppUsage public static TestServiceManager getInstance(@Nullable Context context) { synchronized (TestServiceManager.class) { if (instance == null) { try { instance = new TestServiceManager(ITestService.Stub .asInterface(ServiceManager.getService(Context.TEST_SERVICE)),context); } catch (ServiceNotFoundException e) { throw new IllegalStateException(e); } } return instance; } } public @Nullable String todo() { try { return mService.todo(); } catch (RemoteException e) { throw e.rethrowFromSystemServer(); }}}Copy the code

Can see, in fact, the Manager just Service agent, and the Service is through ServiceManager. GetService to get to, through what we in the SystemServer the addService incoming service_name.

Manager implementation considerations

Higher versions of Android start lint with certain restrictions on managers, such as implementations that must be simple, functions that require annotations to indicate whether arguments and return values can be null, and method names that are case-insensitive (camel) syntactical. Generally, you are advised to implement the problem according to the specifications. If the problem is related to upgrade compatibility, you can ignore the compilation rule according to the compilation log. The specific compilation rule is output to the API_LINt_baseline

Registration Manager

Next, we need to register the Manager in SystemServiceRegistry as we did in the initial analysis, and the user can use getSystemService to get the Manager object and call it. frameworks/base/core/java/android/app/SystemServiceRegistry.java:

static { registerService(Context.TEST_SERVICE, TestServiceManager.class, new CachedServiceFetcher<TestServiceManager>() { @Override public TestServiceManager createService(ContextImpl ctx) { return TestServiceManager.getInstance(ctx); }}); }Copy the code

Supplement current. TXT

Update-api is automatically generated in current. TXT under framework/base/API

SE permissions

Finally, you need to add the SE permission of the service, otherwise you cannot access it. Generally, you need to find the corresponding TE file in the device path to modify it. Here the path is different for each project, and the MTK path under Android source code is taken as an example: device/mediatek/wembley-sepolicy/plat_public/service.te

# ============================================== # MTK Policy Rule # ============================================== # System Server Services # Other Services type nvram_agent_service, service_manager_type; Test_service, app_API_service,system_server_service,service_manager_type;Copy the code

device/mediatek/wembley-sepolicy/plat_private/service_contexts

# ============================================== # MTK Policy Rule # ============================================== # System Server Services # Other Services NvRAMAgent u:object_r:nvram_agent_service:s0 memory_dumper u:object_r:mediaserver_service:s0 imsa u:object_r:radio_service:s0 mtkIms u:object_r:radio_service:s0 GbaService U :object_r:radio_service:s0 # Define our service test u:object_r:test_service:s0Copy the code

use

Since the Service is created by ourselves, the SDK does not expose this interface. Therefore, for use and compilation, we can create a jar package of the Manager class corresponding to the path and provide it to the user, for example: Create the android.app.yy path, create testServicemanager.java, add the empty method todo to it, and compile it into a JAR package for the user. User usage:

/** * Because Lint detects that getSystemService passes static variables to Context, if you want to pass Class, A higher version adaptation is required * so @suppressLint ("WrongConstant") can be ignored */ @SuppressLint("WrongConstant")TestServiceManager Manager = (TestServiceManager) getSystemService("test"); String ret=manager.todo();Copy the code