blob: a8fa4b846bc27dd82c3fa1cc9d58bd716f6c0c9f [file] [log] [blame]
/*
* Copyright (C) 2015 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.internal.telephony;
import android.annotation.NonNull;
import android.content.Context;
import android.content.res.XmlResourceParser;
import android.database.Cursor;
import android.os.Handler;
import android.os.IDeviceIdleController;
import android.os.Looper;
import android.os.ServiceManager;
import android.telephony.Rlog;
import com.android.internal.telephony.cdma.CdmaSubscriptionSourceManager;
import com.android.internal.telephony.cdma.EriManager;
import com.android.internal.telephony.dataconnection.DcTracker;
import com.android.internal.telephony.dataconnection.TransportManager;
import com.android.internal.telephony.emergency.EmergencyNumberTracker;
import com.android.internal.telephony.imsphone.ImsExternalCallTracker;
import com.android.internal.telephony.imsphone.ImsPhone;
import com.android.internal.telephony.imsphone.ImsPhoneCallTracker;
import com.android.internal.telephony.uicc.IccCardStatus;
import com.android.internal.telephony.uicc.UiccCard;
import com.android.internal.telephony.uicc.UiccProfile;
import dalvik.system.PathClassLoader;
import org.xmlpull.v1.XmlPullParser;
import org.xmlpull.v1.XmlPullParserException;
import java.io.IOException;
import java.util.HashSet;
import java.util.Set;
import java.util.function.Consumer;
/**
* This class has one-line methods to instantiate objects only. The purpose is to make code
* unit-test friendly and use this class as a way to do dependency injection. Instantiating objects
* this way makes it easier to mock them in tests.
*/
public class TelephonyComponentFactory {
private static final String TAG = TelephonyComponentFactory.class.getSimpleName();
private static TelephonyComponentFactory sInstance;
private InjectedComponents mInjectedComponents;
private static class InjectedComponents {
private static final String ATTRIBUTE_JAR = "jar";
private static final String ATTRIBUTE_PACKAGE = "package";
private static final String TAG_INJECTION = "injection";
private static final String TAG_COMPONENTS = "components";
private static final String TAG_COMPONENT = "component";
private final Set<String> mComponentNames = new HashSet<>();
private TelephonyComponentFactory mInjectedInstance;
private String mPackageName;
private String mJarPath;
private boolean isInjected() {
return mPackageName != null && mJarPath != null;
}
private void makeInjectedInstance() {
if (isInjected()) {
PathClassLoader classLoader = new PathClassLoader(mJarPath,
ClassLoader.getSystemClassLoader());
try {
Class<?> cls = classLoader.loadClass(mPackageName);
mInjectedInstance = (TelephonyComponentFactory) cls.newInstance();
} catch (ClassNotFoundException e) {
Rlog.e(TAG, "failed: " + e.getMessage());
} catch (IllegalAccessException | InstantiationException e) {
Rlog.e(TAG, "injection failed: " + e.getMessage());
}
}
}
private boolean isComponentInjected(String componentName) {
if (mInjectedInstance == null) {
return false;
}
return mComponentNames.contains(componentName);
}
/**
* Find the injection tag, set attributes, and then parse the injection.
*/
private void parseXml(@NonNull XmlPullParser parser) {
parseXmlByTag(parser, false, p -> {
setAttributes(p);
parseInjection(p);
}, TAG_INJECTION);
}
/**
* Only parse the first injection tag. Find the components tag, then try parse it next.
*/
private void parseInjection(@NonNull XmlPullParser parser) {
parseXmlByTag(parser, false, p -> parseComponents(p), TAG_COMPONENTS);
}
/**
* Only parse the first components tag. Find the component tags, then try parse them next.
*/
private void parseComponents(@NonNull XmlPullParser parser) {
parseXmlByTag(parser, true, p -> parseComponent(p), TAG_COMPONENT);
}
/**
* Extract text values from component tags.
*/
private void parseComponent(@NonNull XmlPullParser parser) {
try {
int outerDepth = parser.getDepth();
int type;
while ((type = parser.next()) != XmlPullParser.END_DOCUMENT
&& (type != XmlPullParser.END_TAG || parser.getDepth() > outerDepth)) {
if (type == XmlPullParser.TEXT) {
mComponentNames.add(parser.getText());
}
}
} catch (XmlPullParserException | IOException e) {
Rlog.e(TAG, "Failed to parse the component." , e);
}
}
/**
* Iterates the tags, finds the corresponding tag and then applies the consumer.
*/
private void parseXmlByTag(@NonNull XmlPullParser parser, boolean allowDuplicate,
@NonNull Consumer<XmlPullParser> consumer, @NonNull final String tag) {
try {
int outerDepth = parser.getDepth();
int type;
while ((type = parser.next()) != XmlPullParser.END_DOCUMENT
&& (type != XmlPullParser.END_TAG || parser.getDepth() > outerDepth)) {
if (type == XmlPullParser.START_TAG && tag.equals(parser.getName())) {
consumer.accept(parser);
if (!allowDuplicate) {
return;
}
}
}
} catch (XmlPullParserException | IOException e) {
Rlog.e(TAG, "Failed to parse or find tag: " + tag, e);
}
}
/**
* Sets the mPackageName and mJarPath by <injection/> tag.
* @param parser
* @return
*/
private void setAttributes(@NonNull XmlPullParser parser) {
for (int i = 0; i < parser.getAttributeCount(); i++) {
String name = parser.getAttributeName(i);
String value = parser.getAttributeValue(i);
if (InjectedComponents.ATTRIBUTE_PACKAGE.equals(name)) {
mPackageName = value;
} else if (InjectedComponents.ATTRIBUTE_JAR.equals(name)) {
mJarPath = value;
}
}
}
}
public static TelephonyComponentFactory getInstance() {
if (sInstance == null) {
sInstance = new TelephonyComponentFactory();
}
return sInstance;
}
/**
* Inject TelephonyComponentFactory using a xml config file.
* @param parser a nullable {@link XmlResourceParser} created with the injection config file.
* The config xml should has below formats:
* <injection package="package.InjectedTelephonyComponentFactory" jar="path to jar file">
* <components>
* <component>example.package.ComponentAbc</component>
* <component>example.package.ComponentXyz</component>
* <!-- e.g. com.android.internal.telephony.GsmCdmaPhone -->
* </components>
* </injection>
*/
public void injectTheComponentFactory(XmlResourceParser parser) {
if (mInjectedComponents != null) {
Rlog.i(TAG, "Already injected.");
return;
}
if (parser != null) {
mInjectedComponents = new InjectedComponents();
mInjectedComponents.parseXml(parser);
mInjectedComponents.makeInjectedInstance();
Rlog.i(TAG, "Total components injected: "
+ mInjectedComponents.mComponentNames.size());
}
}
/**
* Use the injected TelephonyComponentFactory if configured. Otherwise, use the default.
* @param componentName Name of the component class uses the injected component factory,
* e.g. GsmCdmaPhone.class.getName() for {@link GsmCdmaPhone}
* @return injected component factory. If not configured or injected, return the default one.
*/
public TelephonyComponentFactory inject(String componentName) {
if (mInjectedComponents != null && mInjectedComponents.isComponentInjected(componentName)) {
return mInjectedComponents.mInjectedInstance;
}
return sInstance;
}
public GsmCdmaCallTracker makeGsmCdmaCallTracker(GsmCdmaPhone phone) {
return new GsmCdmaCallTracker(phone);
}
public SmsStorageMonitor makeSmsStorageMonitor(Phone phone) {
return new SmsStorageMonitor(phone);
}
public SmsUsageMonitor makeSmsUsageMonitor(Context context) {
return new SmsUsageMonitor(context);
}
public ServiceStateTracker makeServiceStateTracker(GsmCdmaPhone phone, CommandsInterface ci) {
return new ServiceStateTracker(phone, ci);
}
/**
* Create a new EmergencyNumberTracker.
*/
public EmergencyNumberTracker makeEmergencyNumberTracker(Phone phone, CommandsInterface ci) {
return new EmergencyNumberTracker(phone, ci);
}
/**
* Sets the NitzStateMachine implementation to use during implementation. This boolean
* should be removed once the new implementation is stable.
*/
static final boolean USE_NEW_NITZ_STATE_MACHINE = true;
/**
* Returns a new {@link NitzStateMachine} instance.
*/
public NitzStateMachine makeNitzStateMachine(GsmCdmaPhone phone) {
return USE_NEW_NITZ_STATE_MACHINE
? new NewNitzStateMachine(phone)
: new OldNitzStateMachine(phone);
}
public SimActivationTracker makeSimActivationTracker(Phone phone) {
return new SimActivationTracker(phone);
}
public DcTracker makeDcTracker(Phone phone, int transportType) {
return new DcTracker(phone, transportType);
}
public CarrierSignalAgent makeCarrierSignalAgent(Phone phone) {
return new CarrierSignalAgent(phone);
}
public CarrierActionAgent makeCarrierActionAgent(Phone phone) {
return new CarrierActionAgent(phone);
}
public CarrierResolver makeCarrierResolver(Phone phone) {
return new CarrierResolver(phone);
}
public IccPhoneBookInterfaceManager makeIccPhoneBookInterfaceManager(Phone phone) {
return new IccPhoneBookInterfaceManager(phone);
}
public IccSmsInterfaceManager makeIccSmsInterfaceManager(Phone phone) {
return new IccSmsInterfaceManager(phone);
}
/**
* Create a new UiccProfile object.
*/
public UiccProfile makeUiccProfile(Context context, CommandsInterface ci, IccCardStatus ics,
int phoneId, UiccCard uiccCard, Object lock) {
return new UiccProfile(context, ci, ics, phoneId, uiccCard, lock);
}
public EriManager makeEriManager(Phone phone, Context context, int eriFileSource) {
return new EriManager(phone, context, eriFileSource);
}
public WspTypeDecoder makeWspTypeDecoder(byte[] pdu) {
return new WspTypeDecoder(pdu);
}
/**
* Create a tracker for a single-part SMS.
*/
public InboundSmsTracker makeInboundSmsTracker(byte[] pdu, long timestamp, int destPort,
boolean is3gpp2, boolean is3gpp2WapPdu, String address, String displayAddr,
String messageBody) {
return new InboundSmsTracker(pdu, timestamp, destPort, is3gpp2, is3gpp2WapPdu, address,
displayAddr, messageBody);
}
/**
* Create a tracker for a multi-part SMS.
*/
public InboundSmsTracker makeInboundSmsTracker(byte[] pdu, long timestamp, int destPort,
boolean is3gpp2, String address, String displayAddr, int referenceNumber, int sequenceNumber,
int messageCount, boolean is3gpp2WapPdu, String messageBody) {
return new InboundSmsTracker(pdu, timestamp, destPort, is3gpp2, address, displayAddr,
referenceNumber, sequenceNumber, messageCount, is3gpp2WapPdu, messageBody);
}
/**
* Create a tracker from a row of raw table
*/
public InboundSmsTracker makeInboundSmsTracker(Cursor cursor, boolean isCurrentFormat3gpp2) {
return new InboundSmsTracker(cursor, isCurrentFormat3gpp2);
}
public ImsPhoneCallTracker makeImsPhoneCallTracker(ImsPhone imsPhone) {
return new ImsPhoneCallTracker(imsPhone);
}
public ImsExternalCallTracker makeImsExternalCallTracker(ImsPhone imsPhone) {
return new ImsExternalCallTracker(imsPhone);
}
/**
* Create an AppSmsManager for per-app SMS message.
*/
public AppSmsManager makeAppSmsManager(Context context) {
return new AppSmsManager(context);
}
public DeviceStateMonitor makeDeviceStateMonitor(Phone phone) {
return new DeviceStateMonitor(phone);
}
public TransportManager makeTransportManager(Phone phone) {
return new TransportManager(phone);
}
public CdmaSubscriptionSourceManager
getCdmaSubscriptionSourceManagerInstance(Context context, CommandsInterface ci, Handler h,
int what, Object obj) {
return CdmaSubscriptionSourceManager.getInstance(context, ci, h, what, obj);
}
public IDeviceIdleController getIDeviceIdleController() {
return IDeviceIdleController.Stub.asInterface(
ServiceManager.getService(Context.DEVICE_IDLE_CONTROLLER));
}
public LocaleTracker makeLocaleTracker(Phone phone, NitzStateMachine nitzStateMachine,
Looper looper) {
return new LocaleTracker(phone, nitzStateMachine, looper);
}
}