mirror of
https://github.com/mozilla/gecko-dev.git
synced 2024-11-14 15:37:55 +00:00
1655 lines
53 KiB
JavaScript
1655 lines
53 KiB
JavaScript
/* This Source Code Form is subject to the terms of the Mozilla Public
|
|
* License, v. 2.0. If a copy of the MPL was not distributed with this file,
|
|
* You can obtain one at http://mozilla.org/MPL/2.0/. */
|
|
|
|
"use strict";
|
|
|
|
const {classes: Cc, interfaces: Ci, utils: Cu, results: Cr} = Components;
|
|
|
|
Cu.import("resource://gre/modules/XPCOMUtils.jsm");
|
|
Cu.import("resource://gre/modules/Services.jsm");
|
|
Cu.import("resource://gre/modules/systemlibs.js");
|
|
|
|
XPCOMUtils.defineLazyServiceGetter(this, "gSettingsService",
|
|
"@mozilla.org/settingsService;1",
|
|
"nsISettingsService");
|
|
|
|
XPCOMUtils.defineLazyServiceGetter(this, "gNetworkManager",
|
|
"@mozilla.org/network/manager;1",
|
|
"nsINetworkManager");
|
|
|
|
XPCOMUtils.defineLazyServiceGetter(this, "gMobileConnectionService",
|
|
"@mozilla.org/mobileconnection/mobileconnectionservice;1",
|
|
"nsIMobileConnectionService");
|
|
|
|
XPCOMUtils.defineLazyServiceGetter(this, "gIccService",
|
|
"@mozilla.org/icc/iccservice;1",
|
|
"nsIIccService");
|
|
|
|
XPCOMUtils.defineLazyServiceGetter(this, "gDataCallInterfaceService",
|
|
"@mozilla.org/datacall/interfaceservice;1",
|
|
"nsIDataCallInterfaceService");
|
|
|
|
XPCOMUtils.defineLazyGetter(this, "RIL", function() {
|
|
let obj = {};
|
|
Cu.import("resource://gre/modules/ril_consts.js", obj);
|
|
return obj;
|
|
});
|
|
|
|
// Ril quirk to attach data registration on demand.
|
|
let RILQUIRKS_DATA_REGISTRATION_ON_DEMAND =
|
|
libcutils.property_get("ro.moz.ril.data_reg_on_demand", "false") == "true";
|
|
|
|
// Ril quirk to control the uicc/data subscription.
|
|
let RILQUIRKS_SUBSCRIPTION_CONTROL =
|
|
libcutils.property_get("ro.moz.ril.subscription_control", "false") == "true";
|
|
|
|
// Ril quirk to enable IPv6 protocol/roaming protocol in APN settings.
|
|
let RILQUIRKS_HAVE_IPV6 =
|
|
libcutils.property_get("ro.moz.ril.ipv6", "false") == "true";
|
|
|
|
const DATACALLMANAGER_CID =
|
|
Components.ID("{35b9efa2-e42c-45ce-8210-0a13e6f4aadc}");
|
|
const DATACALLHANDLER_CID =
|
|
Components.ID("{132b650f-c4d8-4731-96c5-83785cb31dee}");
|
|
const RILNETWORKINTERFACE_CID =
|
|
Components.ID("{8c11bef9-9b4f-4d96-bed7-f5a1f48eabda}");
|
|
|
|
const TOPIC_XPCOM_SHUTDOWN = "xpcom-shutdown";
|
|
const TOPIC_MOZSETTINGS_CHANGED = "mozsettings-changed";
|
|
const TOPIC_PREF_CHANGED = "nsPref:changed";
|
|
const TOPIC_DATA_CALL_ERROR = "data-call-error";
|
|
const PREF_RIL_DEBUG_ENABLED = "ril.debugging.enabled";
|
|
|
|
const NETWORK_TYPE_UNKNOWN = Ci.nsINetworkInterface.NETWORK_TYPE_UNKNOWN;
|
|
const NETWORK_TYPE_WIFI = Ci.nsINetworkInterface.NETWORK_TYPE_WIFI;
|
|
const NETWORK_TYPE_MOBILE = Ci.nsINetworkInterface.NETWORK_TYPE_MOBILE;
|
|
const NETWORK_TYPE_MOBILE_MMS = Ci.nsINetworkInterface.NETWORK_TYPE_MOBILE_MMS;
|
|
const NETWORK_TYPE_MOBILE_SUPL = Ci.nsINetworkInterface.NETWORK_TYPE_MOBILE_SUPL;
|
|
const NETWORK_TYPE_MOBILE_IMS = Ci.nsINetworkInterface.NETWORK_TYPE_MOBILE_IMS;
|
|
const NETWORK_TYPE_MOBILE_DUN = Ci.nsINetworkInterface.NETWORK_TYPE_MOBILE_DUN;
|
|
|
|
const NETWORK_STATE_UNKNOWN = Ci.nsINetworkInterface.NETWORK_STATE_UNKNOWN;
|
|
const NETWORK_STATE_CONNECTING = Ci.nsINetworkInterface.NETWORK_STATE_CONNECTING;
|
|
const NETWORK_STATE_CONNECTED = Ci.nsINetworkInterface.NETWORK_STATE_CONNECTED;
|
|
const NETWORK_STATE_DISCONNECTING = Ci.nsINetworkInterface.NETWORK_STATE_DISCONNECTING;
|
|
const NETWORK_STATE_DISCONNECTED = Ci.nsINetworkInterface.NETWORK_STATE_DISCONNECTED;
|
|
|
|
const INT32_MAX = 2147483647;
|
|
|
|
// set to true in ril_consts.js to see debug messages
|
|
var DEBUG = RIL.DEBUG_RIL;
|
|
|
|
function updateDebugFlag() {
|
|
// Read debug setting from pref
|
|
let debugPref;
|
|
try {
|
|
debugPref = Services.prefs.getBoolPref(PREF_RIL_DEBUG_ENABLED);
|
|
} catch (e) {
|
|
debugPref = false;
|
|
}
|
|
DEBUG = debugPref || RIL.DEBUG_RIL;
|
|
}
|
|
updateDebugFlag();
|
|
|
|
function DataCallManager() {
|
|
this._connectionHandlers = [];
|
|
|
|
let numRadioInterfaces = gMobileConnectionService.numItems;
|
|
for (let clientId = 0; clientId < numRadioInterfaces; clientId++) {
|
|
this._connectionHandlers.push(new DataCallHandler(clientId));
|
|
}
|
|
|
|
let lock = gSettingsService.createLock();
|
|
// Read the APN data from the settings DB.
|
|
lock.get("ril.data.apnSettings", this);
|
|
// Read the data enabled setting from DB.
|
|
lock.get("ril.data.enabled", this);
|
|
lock.get("ril.data.roaming_enabled", this);
|
|
// Read the default client id for data call.
|
|
lock.get("ril.data.defaultServiceId", this);
|
|
|
|
Services.obs.addObserver(this, TOPIC_XPCOM_SHUTDOWN, false);
|
|
Services.obs.addObserver(this, TOPIC_MOZSETTINGS_CHANGED, false);
|
|
Services.prefs.addObserver(PREF_RIL_DEBUG_ENABLED, this, false);
|
|
}
|
|
DataCallManager.prototype = {
|
|
classID: DATACALLMANAGER_CID,
|
|
classInfo: XPCOMUtils.generateCI({classID: DATACALLMANAGER_CID,
|
|
classDescription: "Data Call Manager",
|
|
interfaces: [Ci.nsIDataCallManager]}),
|
|
|
|
QueryInterface: XPCOMUtils.generateQI([Ci.nsIDataCallManager,
|
|
Ci.nsIObserver,
|
|
Ci.nsISettingsServiceCallback]),
|
|
|
|
_connectionHandlers: null,
|
|
|
|
// Flag to determine the data state to start with when we boot up. It
|
|
// corresponds to the 'ril.data.enabled' setting from the UI.
|
|
_dataEnabled: false,
|
|
|
|
// Flag to record the default client id for data call. It corresponds to
|
|
// the 'ril.data.defaultServiceId' setting from the UI.
|
|
_dataDefaultClientId: -1,
|
|
|
|
// Flag to record the current default client id for data call.
|
|
// It differs from _dataDefaultClientId in that it is set only when
|
|
// the switch of client id process is done.
|
|
_currentDataClientId: -1,
|
|
|
|
// Pending function to execute when we are notified that another data call has
|
|
// been disconnected.
|
|
_pendingDataCallRequest: null,
|
|
|
|
debug: function(aMsg) {
|
|
dump("-*- DataCallManager: " + aMsg + "\n");
|
|
},
|
|
|
|
getDataCallHandler: function(aClientId) {
|
|
let handler = this._connectionHandlers[aClientId]
|
|
if (!handler) {
|
|
throw Cr.NS_ERROR_UNEXPECTED;
|
|
}
|
|
|
|
return handler;
|
|
},
|
|
|
|
_setDataRegistration: function(aDataCallInterface, aAttach) {
|
|
return new Promise(function(aResolve, aReject) {
|
|
let callback = {
|
|
QueryInterface: XPCOMUtils.generateQI([Ci.nsIDataCallCallback]),
|
|
notifySuccess: function() {
|
|
aResolve();
|
|
},
|
|
notifyError: function(aErrorMsg) {
|
|
aReject(aErrorMsg);
|
|
}
|
|
};
|
|
|
|
aDataCallInterface.setDataRegistration(aAttach, callback);
|
|
});
|
|
},
|
|
|
|
_handleDataClientIdChange: function(aNewClientId) {
|
|
if (this._dataDefaultClientId === aNewClientId) {
|
|
return;
|
|
}
|
|
this._dataDefaultClientId = aNewClientId;
|
|
|
|
// This is to handle boot up stage.
|
|
if (this._currentDataClientId == -1) {
|
|
this._currentDataClientId = this._dataDefaultClientId;
|
|
let connHandler = this._connectionHandlers[this._currentDataClientId];
|
|
let dcInterface = connHandler.dataCallInterface;
|
|
if (RILQUIRKS_DATA_REGISTRATION_ON_DEMAND ||
|
|
RILQUIRKS_SUBSCRIPTION_CONTROL) {
|
|
this._setDataRegistration(dcInterface, true);
|
|
}
|
|
if (this._dataEnabled) {
|
|
let settings = connHandler.dataCallSettings;
|
|
settings.oldEnabled = settings.enabled;
|
|
settings.enabled = true;
|
|
connHandler.updateRILNetworkInterface();
|
|
}
|
|
return;
|
|
}
|
|
|
|
let oldConnHandler = this._connectionHandlers[this._currentDataClientId];
|
|
let oldIface = oldConnHandler.dataCallInterface;
|
|
let oldSettings = oldConnHandler.dataCallSettings;
|
|
let newConnHandler = this._connectionHandlers[this._dataDefaultClientId];
|
|
let newIface = newConnHandler.dataCallInterface;
|
|
let newSettings = newConnHandler.dataCallSettings;
|
|
|
|
let applyPendingDataSettings = () => {
|
|
if (DEBUG) {
|
|
this.debug("Apply pending data registration and settings.");
|
|
}
|
|
|
|
if (RILQUIRKS_DATA_REGISTRATION_ON_DEMAND ||
|
|
RILQUIRKS_SUBSCRIPTION_CONTROL) {
|
|
this._setDataRegistration(oldIface, false).then(() => {
|
|
if (this._dataEnabled) {
|
|
newSettings.oldEnabled = newSettings.enabled;
|
|
newSettings.enabled = true;
|
|
}
|
|
this._currentDataClientId = this._dataDefaultClientId;
|
|
|
|
this._setDataRegistration(newIface, true).then(() => {
|
|
newConnHandler.updateRILNetworkInterface();
|
|
});
|
|
});
|
|
return;
|
|
}
|
|
|
|
if (this._dataEnabled) {
|
|
newSettings.oldEnabled = newSettings.enabled;
|
|
newSettings.enabled = true;
|
|
}
|
|
|
|
this._currentDataClientId = this._dataDefaultClientId;
|
|
newConnHandler.updateRILNetworkInterface();
|
|
};
|
|
|
|
if (this._dataEnabled) {
|
|
oldSettings.oldEnabled = oldSettings.enabled;
|
|
oldSettings.enabled = false;
|
|
}
|
|
|
|
oldConnHandler.deactivateDataCallsAndWait().then(() => {
|
|
applyPendingDataSettings();
|
|
});
|
|
},
|
|
|
|
_shutdown: function() {
|
|
for (let handler of this._connectionHandlers) {
|
|
handler.shutdown();
|
|
}
|
|
this._connectionHandlers = null;
|
|
Services.prefs.removeObserver(PREF_RIL_DEBUG_ENABLED, this);
|
|
Services.obs.removeObserver(this, TOPIC_XPCOM_SHUTDOWN);
|
|
Services.obs.removeObserver(this, TOPIC_MOZSETTINGS_CHANGED);
|
|
},
|
|
|
|
/**
|
|
* nsISettingsServiceCallback
|
|
*/
|
|
handle: function(aName, aResult) {
|
|
switch (aName) {
|
|
case "ril.data.apnSettings":
|
|
if (DEBUG) {
|
|
this.debug("'ril.data.apnSettings' is now " +
|
|
JSON.stringify(aResult));
|
|
}
|
|
if (!aResult) {
|
|
break;
|
|
}
|
|
for (let clientId in this._connectionHandlers) {
|
|
let handler = this._connectionHandlers[clientId];
|
|
let apnSetting = aResult[clientId];
|
|
if (handler && apnSetting) {
|
|
handler.updateApnSettings(apnSetting);
|
|
}
|
|
}
|
|
break;
|
|
case "ril.data.enabled":
|
|
if (DEBUG) {
|
|
this.debug("'ril.data.enabled' is now " + aResult);
|
|
}
|
|
if (this._dataEnabled === aResult) {
|
|
break;
|
|
}
|
|
this._dataEnabled = aResult;
|
|
|
|
if (DEBUG) {
|
|
this.debug("Default id for data call: " + this._dataDefaultClientId);
|
|
}
|
|
if (this._dataDefaultClientId === -1) {
|
|
// We haven't got the default id for data from db.
|
|
break;
|
|
}
|
|
|
|
let connHandler = this._connectionHandlers[this._dataDefaultClientId];
|
|
let settings = connHandler.dataCallSettings;
|
|
settings.oldEnabled = settings.enabled;
|
|
settings.enabled = aResult;
|
|
connHandler.updateRILNetworkInterface();
|
|
break;
|
|
case "ril.data.roaming_enabled":
|
|
if (DEBUG) {
|
|
this.debug("'ril.data.roaming_enabled' is now " + aResult);
|
|
this.debug("Default id for data call: " + this._dataDefaultClientId);
|
|
}
|
|
for (let clientId = 0; clientId < this._connectionHandlers.length; clientId++) {
|
|
let connHandler = this._connectionHandlers[clientId];
|
|
let settings = connHandler.dataCallSettings;
|
|
settings.roamingEnabled = Array.isArray(aResult) ? aResult[clientId]
|
|
: aResult;
|
|
}
|
|
if (this._dataDefaultClientId === -1) {
|
|
// We haven't got the default id for data from db.
|
|
break;
|
|
}
|
|
this._connectionHandlers[this._dataDefaultClientId].updateRILNetworkInterface();
|
|
break;
|
|
case "ril.data.defaultServiceId":
|
|
aResult = aResult || 0;
|
|
if (DEBUG) {
|
|
this.debug("'ril.data.defaultServiceId' is now " + aResult);
|
|
}
|
|
this._handleDataClientIdChange(aResult);
|
|
break;
|
|
}
|
|
},
|
|
|
|
handleError: function(aErrorMessage) {
|
|
if (DEBUG) {
|
|
this.debug("There was an error while reading RIL settings.");
|
|
}
|
|
},
|
|
|
|
/**
|
|
* nsIObserver interface methods.
|
|
*/
|
|
observe: function(aSubject, aTopic, aData) {
|
|
switch (aTopic) {
|
|
case TOPIC_MOZSETTINGS_CHANGED:
|
|
if ("wrappedJSObject" in aSubject) {
|
|
aSubject = aSubject.wrappedJSObject;
|
|
}
|
|
this.handle(aSubject.key, aSubject.value);
|
|
break;
|
|
case TOPIC_PREF_CHANGED:
|
|
if (aData === PREF_RIL_DEBUG_ENABLED) {
|
|
updateDebugFlag();
|
|
}
|
|
break;
|
|
case TOPIC_XPCOM_SHUTDOWN:
|
|
this._shutdown();
|
|
break;
|
|
}
|
|
},
|
|
};
|
|
|
|
function DataCallHandler(aClientId) {
|
|
// Initial owning attributes.
|
|
this.clientId = aClientId;
|
|
this.dataCallSettings = {
|
|
oldEnabled: false,
|
|
enabled: false,
|
|
roamingEnabled: false
|
|
};
|
|
this._dataCalls = [];
|
|
this._listeners = [];
|
|
|
|
// This map is used to collect all the apn types and its corresponding
|
|
// RILNetworkInterface.
|
|
this.dataNetworkInterfaces = new Map();
|
|
|
|
this.dataCallInterface = gDataCallInterfaceService.getDataCallInterface(aClientId);
|
|
this.dataCallInterface.registerListener(this);
|
|
|
|
let mobileConnection = gMobileConnectionService.getItemByServiceId(aClientId);
|
|
mobileConnection.registerListener(this);
|
|
|
|
this._dataInfo = {
|
|
state: mobileConnection.data.state,
|
|
type: mobileConnection.data.type,
|
|
roaming: mobileConnection.data.roaming
|
|
}
|
|
}
|
|
DataCallHandler.prototype = {
|
|
classID: DATACALLHANDLER_CID,
|
|
classInfo: XPCOMUtils.generateCI({classID: DATACALLHANDLER_CID,
|
|
classDescription: "Data Call Handler",
|
|
interfaces: [Ci.nsIDataCallHandler]}),
|
|
|
|
QueryInterface: XPCOMUtils.generateQI([Ci.nsIDataCallHandler,
|
|
Ci.nsIDataCallInterfaceListener,
|
|
Ci.nsIMobileConnectionListener]),
|
|
|
|
clientId: 0,
|
|
dataCallInterface: null,
|
|
dataCallSettings: null,
|
|
dataNetworkInterfaces: null,
|
|
_dataCalls: null,
|
|
_dataInfo: null,
|
|
|
|
// Apn settings to be setup after data call are cleared.
|
|
_pendingApnSettings: null,
|
|
|
|
debug: function(aMsg) {
|
|
dump("-*- DataCallHandler[" + this.clientId + "]: " + aMsg + "\n");
|
|
},
|
|
|
|
shutdown: function() {
|
|
// Shutdown all RIL network interfaces
|
|
this.dataNetworkInterfaces.forEach(function(networkInterface) {
|
|
gNetworkManager.unregisterNetworkInterface(networkInterface);
|
|
networkInterface.shutdown();
|
|
networkInterface = null;
|
|
});
|
|
this.dataNetworkInterfaces.clear();
|
|
this._dataCalls = [];
|
|
this.clientId = null;
|
|
|
|
this.dataCallinterface.unregisterListener(this);
|
|
this.dataCallInterface = null;
|
|
|
|
let mobileConnection =
|
|
gMobileConnectionService.getItemByServiceId(this.clientId);
|
|
mobileConnection.unregisterListener(this);
|
|
},
|
|
|
|
/**
|
|
* Check if we get all necessary APN data.
|
|
*/
|
|
_validateApnSetting: function(aApnSetting) {
|
|
return (aApnSetting &&
|
|
aApnSetting.apn &&
|
|
aApnSetting.types &&
|
|
aApnSetting.types.length);
|
|
},
|
|
|
|
_convertApnType: function(aApnType) {
|
|
switch (aApnType) {
|
|
case "default":
|
|
return NETWORK_TYPE_MOBILE;
|
|
case "mms":
|
|
return NETWORK_TYPE_MOBILE_MMS;
|
|
case "supl":
|
|
return NETWORK_TYPE_MOBILE_SUPL;
|
|
case "ims":
|
|
return NETWORK_TYPE_MOBILE_IMS;
|
|
case "dun":
|
|
return NETWORK_TYPE_MOBILE_DUN;
|
|
default:
|
|
return NETWORK_TYPE_UNKNOWN;
|
|
}
|
|
},
|
|
|
|
_compareDataCallOptions: function(aDataCall, aNewDataCall) {
|
|
return aDataCall.apnProfile.apn == aNewDataCall.apnProfile.apn &&
|
|
aDataCall.apnProfile.user == aNewDataCall.apnProfile.user &&
|
|
aDataCall.apnProfile.password == aNewDataCall.apnProfile.passwd &&
|
|
aDataCall.apnProfile.authType == aNewDataCall.apnProfile.authType &&
|
|
aDataCall.apnProfile.protocol == aNewDataCall.apnProfile.protocol &&
|
|
aDataCall.apnProfile.roaming_protocol == aNewDataCall.apnProfile.roaming_protocol;
|
|
},
|
|
|
|
/**
|
|
* This function will do the following steps:
|
|
* 1. Clear the cached APN settings in the RIL.
|
|
* 2. Combine APN, user name, and password as the key of |byApn| object to
|
|
* refer to the corresponding APN setting.
|
|
* 3. Use APN type as the index of |byType| object to refer to the
|
|
* corresponding APN setting.
|
|
* 4. Create RilNetworkInterface for each APN setting created at step 2.
|
|
*/
|
|
_setupApnSettings: function(aNewApnSettings) {
|
|
if (!aNewApnSettings) {
|
|
return;
|
|
}
|
|
if (DEBUG) this.debug("setupApnSettings: " + JSON.stringify(aNewApnSettings));
|
|
|
|
// Shutdown all network interfaces and clear data calls.
|
|
this.dataNetworkInterfaces.forEach(function(networkInterface) {
|
|
gNetworkManager.unregisterNetworkInterface(networkInterface);
|
|
networkInterface.shutdown();
|
|
networkInterface = null;
|
|
});
|
|
this.dataNetworkInterfaces.clear();
|
|
this._dataCalls = [];
|
|
|
|
// Cache the APN settings by APNs and by types in the RIL.
|
|
for (let inputApnSetting of aNewApnSettings) {
|
|
if (!this._validateApnSetting(inputApnSetting)) {
|
|
continue;
|
|
}
|
|
|
|
// Use APN type as the key of dataNetworkInterfaces to refer to the
|
|
// corresponding RILNetworkInterface.
|
|
for (let i = 0; i < inputApnSetting.types.length; i++) {
|
|
let apnType = inputApnSetting.types[i];
|
|
let networkType = this._convertApnType(apnType);
|
|
if (networkType === NETWORK_TYPE_UNKNOWN) {
|
|
if (DEBUG) this.debug("Invalid apn type: " + apnType);
|
|
continue;
|
|
}
|
|
|
|
if (DEBUG) this.debug("Preparing RILNetworkInterface for type: " + apnType);
|
|
// Create DataCall for RILNetworkInterface or reuse one that is shareable.
|
|
let dataCall;
|
|
for (let i = 0; i < this._dataCalls.length; i++) {
|
|
if (this._dataCalls[i].canHandleApn(inputApnSetting)) {
|
|
if (DEBUG) this.debug("Found shareable DataCall, reusing it.");
|
|
dataCall = this._dataCalls[i];
|
|
break;
|
|
}
|
|
}
|
|
|
|
if (!dataCall) {
|
|
if (DEBUG) this.debug("No shareable DataCall found, creating one.");
|
|
dataCall = new DataCall(this.clientId, inputApnSetting, this);
|
|
this._dataCalls.push(dataCall);
|
|
}
|
|
|
|
try {
|
|
let networkInterface = new RILNetworkInterface(this, networkType,
|
|
inputApnSetting,
|
|
dataCall);
|
|
gNetworkManager.registerNetworkInterface(networkInterface);
|
|
this.dataNetworkInterfaces.set(networkType, networkInterface);
|
|
} catch (e) {
|
|
if (DEBUG) {
|
|
this.debug("Error setting up RILNetworkInterface for type " +
|
|
apnType + ": " + e);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
},
|
|
|
|
/**
|
|
* Check if all data is disconnected.
|
|
*/
|
|
allDataDisconnected: function() {
|
|
for (let i = 0; i < this._dataCalls.length; i++) {
|
|
let dataCall = this._dataCalls[i];
|
|
if (dataCall.state != NETWORK_STATE_UNKNOWN &&
|
|
dataCall.state != NETWORK_STATE_DISCONNECTED) {
|
|
return false;
|
|
}
|
|
}
|
|
return true;
|
|
},
|
|
|
|
deactivateDataCallsAndWait: function() {
|
|
return new Promise((aResolve, aReject) => {
|
|
this.deactivateDataCalls({
|
|
notifyDataCallsDisconnected: function() {
|
|
aResolve();
|
|
}
|
|
});
|
|
});
|
|
},
|
|
|
|
updateApnSettings: function(aNewApnSettings) {
|
|
if (!aNewApnSettings) {
|
|
return;
|
|
}
|
|
if (this._pendingApnSettings) {
|
|
// Change of apn settings in process, just update to the newest.
|
|
this._pengingApnSettings = aNewApnSettings;
|
|
return;
|
|
}
|
|
|
|
this._pendingApnSettings = aNewApnSettings;
|
|
this.deactivateDataCallsAndWait().then(() => {
|
|
this._setupApnSettings(this._pendingApnSettings);
|
|
this._pendingApnSettings = null;
|
|
this.updateRILNetworkInterface();
|
|
});
|
|
},
|
|
|
|
updateRILNetworkInterface: function() {
|
|
let networkInterface = this.dataNetworkInterfaces.get(NETWORK_TYPE_MOBILE);
|
|
if (!networkInterface) {
|
|
if (DEBUG) {
|
|
this.debug("No network interface for default data.");
|
|
}
|
|
return;
|
|
}
|
|
|
|
let connection =
|
|
gMobileConnectionService.getItemByServiceId(this.clientId);
|
|
|
|
// This check avoids data call connection if the radio is not ready
|
|
// yet after toggling off airplane mode.
|
|
let radioState = connection && connection.radioState;
|
|
if (radioState != Ci.nsIMobileConnection.MOBILE_RADIO_STATE_ENABLED) {
|
|
if (DEBUG) {
|
|
this.debug("RIL is not ready for data connection: radio's not ready");
|
|
}
|
|
return;
|
|
}
|
|
|
|
// We only watch at "ril.data.enabled" flag changes for connecting or
|
|
// disconnecting the data call. If the value of "ril.data.enabled" is
|
|
// true and any of the remaining flags change the setting application
|
|
// should turn this flag to false and then to true in order to reload
|
|
// the new values and reconnect the data call.
|
|
if (this.dataCallSettings.oldEnabled === this.dataCallSettings.enabled) {
|
|
if (DEBUG) {
|
|
this.debug("No changes for ril.data.enabled flag. Nothing to do.");
|
|
}
|
|
return;
|
|
}
|
|
|
|
let dataInfo = connection && connection.data;
|
|
let isRegistered =
|
|
dataInfo &&
|
|
dataInfo.state == RIL.GECKO_MOBILE_CONNECTION_STATE_REGISTERED;
|
|
let haveDataConnection =
|
|
dataInfo &&
|
|
dataInfo.type != RIL.GECKO_MOBILE_CONNECTION_STATE_UNKNOWN;
|
|
if (!isRegistered || !haveDataConnection) {
|
|
if (DEBUG) {
|
|
this.debug("RIL is not ready for data connection: Phone's not " +
|
|
"registered or doesn't have data connection.");
|
|
}
|
|
return;
|
|
}
|
|
let wifi_active = false;
|
|
if (gNetworkManager.active &&
|
|
gNetworkManager.active.type == NETWORK_TYPE_WIFI) {
|
|
wifi_active = true;
|
|
}
|
|
|
|
let defaultDataCallConnected = networkInterface.connected;
|
|
|
|
// We have moved part of the decision making into DataCall, the rest will be
|
|
// moved after Bug 904514 - [meta] NetworkManager enhancement.
|
|
if (networkInterface.enabled &&
|
|
(!this.dataCallSettings.enabled ||
|
|
(dataInfo.roaming && !this.dataCallSettings.roamingEnabled))) {
|
|
if (DEBUG) {
|
|
this.debug("Data call settings: disconnect data call.");
|
|
}
|
|
networkInterface.disconnect();
|
|
return;
|
|
}
|
|
|
|
if (networkInterface.enabled && wifi_active) {
|
|
if (DEBUG) {
|
|
this.debug("Disconnect data call when Wifi is connected.");
|
|
}
|
|
networkInterface.disconnect();
|
|
return;
|
|
}
|
|
|
|
if (!this.dataCallSettings.enabled || defaultDataCallConnected) {
|
|
if (DEBUG) {
|
|
this.debug("Data call settings: nothing to do.");
|
|
}
|
|
return;
|
|
}
|
|
if (dataInfo.roaming && !this.dataCallSettings.roamingEnabled) {
|
|
if (DEBUG) {
|
|
this.debug("We're roaming, but data roaming is disabled.");
|
|
}
|
|
return;
|
|
}
|
|
if (wifi_active) {
|
|
if (DEBUG) {
|
|
this.debug("Don't connect data call when Wifi is connected.");
|
|
}
|
|
return;
|
|
}
|
|
if (this._pendingApnSettings) {
|
|
if (DEBUG) this.debug("We're changing apn settings, ignore any changes.");
|
|
return;
|
|
}
|
|
|
|
if (this._deactivatingDataCalls) {
|
|
if (DEBUG) this.debug("We're deactivating all data calls, ignore any changes.");
|
|
return;
|
|
}
|
|
|
|
if (DEBUG) {
|
|
this.debug("Data call settings: connect data call.");
|
|
}
|
|
networkInterface.connect();
|
|
},
|
|
|
|
_isMobileNetworkType: function(aNetworkType) {
|
|
if (aNetworkType === NETWORK_TYPE_MOBILE ||
|
|
aNetworkType === NETWORK_TYPE_MOBILE_MMS ||
|
|
aNetworkType === NETWORK_TYPE_MOBILE_SUPL ||
|
|
aNetworkType === NETWORK_TYPE_MOBILE_IMS ||
|
|
aNetworkType === NETWORK_TYPE_MOBILE_DUN) {
|
|
return true;
|
|
}
|
|
|
|
return false;
|
|
},
|
|
|
|
getDataCallStateByType: function(aNetworkType) {
|
|
if (!this._isMobileNetworkType(aNetworkType)) {
|
|
if (DEBUG) this.debug(aNetworkType + " is not a mobile network type!");
|
|
throw Cr.NS_ERROR_INVALID_ARG;
|
|
}
|
|
|
|
let networkInterface = this.dataNetworkInterfaces.get(aNetworkType);
|
|
if (!networkInterface) {
|
|
return NETWORK_STATE_UNKNOWN;
|
|
}
|
|
return networkInterface.state;
|
|
},
|
|
|
|
setupDataCallByType: function(aNetworkType) {
|
|
if (DEBUG) {
|
|
this.debug("setupDataCallByType: " + aNetworkType);
|
|
}
|
|
|
|
if (!this._isMobileNetworkType(aNetworkType)) {
|
|
if (DEBUG) this.debug(aNetworkType + " is not a mobile network type!");
|
|
throw Cr.NS_ERROR_INVALID_ARG;
|
|
}
|
|
|
|
let networkInterface = this.dataNetworkInterfaces.get(aNetworkType);
|
|
if (!networkInterface) {
|
|
if (DEBUG) {
|
|
this.debug("No network interface for type: " + aNetworkType);
|
|
}
|
|
return;
|
|
}
|
|
|
|
networkInterface.connect();
|
|
},
|
|
|
|
deactivateDataCallByType: function(aNetworkType) {
|
|
if (DEBUG) {
|
|
this.debug("deactivateDataCallByType: " + aNetworkType);
|
|
}
|
|
|
|
if (!this._isMobileNetworkType(aNetworkType)) {
|
|
if (DEBUG) this.debug(aNetworkType + " is not a mobile network type!");
|
|
throw Cr.NS_ERROR_INVALID_ARG;
|
|
}
|
|
|
|
let networkInterface = this.dataNetworkInterfaces.get(aNetworkType);
|
|
if (!networkInterface) {
|
|
if (DEBUG) {
|
|
this.debug("No network interface for type: " + aNetworkType);
|
|
}
|
|
return;
|
|
}
|
|
|
|
networkInterface.disconnect();
|
|
},
|
|
|
|
_deactivatingDataCalls: false,
|
|
|
|
deactivateDataCalls: function(aCallback) {
|
|
let dataDisconnecting = false;
|
|
this.dataNetworkInterfaces.forEach(function(networkInterface) {
|
|
if (networkInterface.enabled) {
|
|
if (networkInterface.state != NETWORK_STATE_UNKNOWN &&
|
|
networkInterface.state != NETWORK_STATE_DISCONNECTED) {
|
|
dataDisconnecting = true;
|
|
}
|
|
networkInterface.disconnect();
|
|
}
|
|
});
|
|
|
|
this._deactivatingDataCalls = dataDisconnecting;
|
|
if (!dataDisconnecting) {
|
|
aCallback.notifyDataCallsDisconnected();
|
|
return;
|
|
}
|
|
|
|
let callback = {
|
|
notifyAllDataDisconnected: () => {
|
|
this._unregisterListener(callback);
|
|
aCallback.notifyDataCallsDisconnected();
|
|
}
|
|
};
|
|
this._registerListener(callback);
|
|
},
|
|
|
|
_listeners: null,
|
|
|
|
_notifyListeners: function(aMethodName, aArgs) {
|
|
let listeners = this._listeners.slice();
|
|
for (let listener of listeners) {
|
|
if (this._listeners.indexOf(listener) == -1) {
|
|
// Listener has been unregistered in previous run.
|
|
continue;
|
|
}
|
|
|
|
let handler = listener[aMethodName];
|
|
try {
|
|
handler.apply(listener, aArgs);
|
|
} catch (e) {
|
|
this.debug("listener for " + aMethodName + " threw an exception: " + e);
|
|
}
|
|
}
|
|
},
|
|
|
|
_registerListener: function(aListener) {
|
|
if (this._listeners.indexOf(aListener) >= 0) {
|
|
return;
|
|
}
|
|
|
|
this._listeners.push(aListener);
|
|
},
|
|
|
|
_unregisterListener: function(aListener) {
|
|
let index = this._listeners.indexOf(aListener);
|
|
if (index >= 0) {
|
|
this._listeners.splice(index, 1);
|
|
}
|
|
},
|
|
|
|
_findDataCallByCid: function(aCid) {
|
|
if (aCid === undefined || aCid < 0) {
|
|
return -1;
|
|
}
|
|
|
|
for (let i = 0; i < this._dataCalls.length; i++) {
|
|
let datacall = this._dataCalls[i];
|
|
if (datacall.linkInfo.cid != null &&
|
|
datacall.linkInfo.cid == aCid) {
|
|
return i;
|
|
}
|
|
}
|
|
|
|
return -1;
|
|
},
|
|
|
|
/**
|
|
* Notify about data call setup error, called from DataCall.
|
|
*/
|
|
notifyDataCallError: function(aMessage) {
|
|
// Notify data call error only for data APN
|
|
let networkInterface = this.dataNetworkInterfaces.get(NETWORK_TYPE_MOBILE);
|
|
if (networkInterface && networkInterface.enabled) {
|
|
let dataCall = networkInterface.dataCall;
|
|
// If there is a cid, compare cid; otherwise it is probably an error on
|
|
// data call setup.
|
|
if (aMessage.cid !== undefined) {
|
|
if (aMessage.linkInfo.cid == dataCall.linkInfo.cid) {
|
|
Services.obs.notifyObservers(networkInterface, TOPIC_DATA_CALL_ERROR,
|
|
null);
|
|
}
|
|
} else {
|
|
if (this._compareDataCallOptions(dataCall, aMessage)) {
|
|
Services.obs.notifyObservers(networkInterface, TOPIC_DATA_CALL_ERROR,
|
|
null);
|
|
}
|
|
}
|
|
}
|
|
},
|
|
|
|
/**
|
|
* Notify about data call changed, called from DataCall.
|
|
*/
|
|
notifyDataCallChanged: function(aUpdatedDataCall) {
|
|
// Process pending radio power off request after all data calls
|
|
// are disconnected.
|
|
if (aUpdatedDataCall.state == NETWORK_STATE_DISCONNECTED ||
|
|
aUpdatedDataCall.state == NETWORK_STATE_UNKNOWN &&
|
|
this.allDataDisconnected() && this._deactivatingDataCalls) {
|
|
this._deactivatingDataCalls = false;
|
|
this._notifyListeners("notifyAllDataDisconnected", {
|
|
clientId: this.clientId
|
|
});
|
|
}
|
|
},
|
|
|
|
// nsIDataCallInterfaceListener
|
|
|
|
notifyDataCallListChanged: function(aCount, aDataCallList) {
|
|
let currentDataCalls = this._dataCalls.slice();
|
|
for (let i = 0; i < aDataCallList.length; i++) {
|
|
let dataCall = aDataCallList[i];
|
|
let index = this._findDataCallByCid(dataCall.cid);
|
|
if (index == -1) {
|
|
if (DEBUG) {
|
|
this.debug("Unexpected new data call: " + JSON.stringify(dataCall));
|
|
}
|
|
continue;
|
|
}
|
|
currentDataCalls[index].onDataCallChanged(dataCall);
|
|
currentDataCalls[index] = null;
|
|
}
|
|
|
|
// If there is any CONNECTED DataCall left in currentDataCalls, means that
|
|
// it is missing in dataCallList, we should send a DISCONNECTED event to
|
|
// notify about this.
|
|
for (let i = 0; i < currentDataCalls.length; i++) {
|
|
let currentDataCall = currentDataCalls[i];
|
|
if (currentDataCall && currentDataCall.linkInfo.cid != null &&
|
|
currentDataCall.state == NETWORK_STATE_CONNECTED) {
|
|
if (DEBUG) {
|
|
this.debug("Expected data call missing: " + JSON.stringify(
|
|
currentDataCall.apnProfile) + ", must have been DISCONNECTED.");
|
|
}
|
|
currentDataCall.onDataCallChanged({
|
|
state: NETWORK_STATE_DISCONNECTED
|
|
});
|
|
}
|
|
}
|
|
},
|
|
|
|
// nsIMobileConnectionListener
|
|
|
|
notifyVoiceChanged: function() {},
|
|
|
|
notifyDataChanged: function () {
|
|
let connection = gMobileConnectionService.getItemByServiceId(this.clientId);
|
|
let newDataInfo = connection.data;
|
|
|
|
if (this._dataInfo.state == newDataInfo.state &&
|
|
this._dataInfo.type == newDataInfo.type &&
|
|
this._dataInfo.roaming == newDataInfo.roaming) {
|
|
return;
|
|
}
|
|
|
|
this._dataInfo.state = newDataInfo.state;
|
|
this._dataInfo.type = newDataInfo.type;
|
|
this._dataInfo.roaming = newDataInfo.roaming;
|
|
this.updateRILNetworkInterface();
|
|
},
|
|
|
|
notifyDataError: function (aMessage) {},
|
|
|
|
notifyCFStateChanged: function(aAction, aReason, aNumber, aTimeSeconds, aServiceClass) {},
|
|
|
|
notifyEmergencyCbModeChanged: function(aActive, aTimeoutMs) {},
|
|
|
|
notifyOtaStatusChanged: function(aStatus) {},
|
|
|
|
notifyRadioStateChanged: function() {},
|
|
|
|
notifyClirModeChanged: function(aMode) {},
|
|
|
|
notifyLastKnownNetworkChanged: function() {},
|
|
|
|
notifyLastKnownHomeNetworkChanged: function() {},
|
|
|
|
notifyNetworkSelectionModeChanged: function() {}
|
|
};
|
|
|
|
function DataCall(aClientId, aApnSetting, aDataCallHandler) {
|
|
this.clientId = aClientId;
|
|
this.dataCallHandler = aDataCallHandler;
|
|
this.apnProfile = {
|
|
apn: aApnSetting.apn,
|
|
user: aApnSetting.user,
|
|
password: aApnSetting.password,
|
|
authType: aApnSetting.authtype,
|
|
protocol: aApnSetting.protocol,
|
|
roaming_protocol: aApnSetting.roaming_protocol
|
|
};
|
|
this.linkInfo = {
|
|
cid: null,
|
|
ifname: null,
|
|
addresses: [],
|
|
dnses: [],
|
|
gateways: []
|
|
};
|
|
this.state = NETWORK_STATE_UNKNOWN;
|
|
this.requestedNetworkIfaces = [];
|
|
}
|
|
DataCall.prototype = {
|
|
/**
|
|
* Standard values for the APN connection retry process
|
|
* Retry funcion: time(secs) = A * numer_of_retries^2 + B
|
|
*/
|
|
NETWORK_APNRETRY_FACTOR: 8,
|
|
NETWORK_APNRETRY_ORIGIN: 3,
|
|
NETWORK_APNRETRY_MAXRETRIES: 10,
|
|
|
|
dataCallHandler: null,
|
|
|
|
// Event timer for connection retries
|
|
timer: null,
|
|
|
|
// APN failed connections. Retry counter
|
|
apnRetryCounter: 0,
|
|
|
|
// Array to hold RILNetworkInterfaces that requested this DataCall.
|
|
requestedNetworkIfaces: null,
|
|
|
|
/**
|
|
* @return "deactivate" if <ifname> changes or one of the aCurrentDataCall
|
|
* addresses is missing in updatedDataCall, or "identical" if no
|
|
* changes found, or "changed" otherwise.
|
|
*/
|
|
_compareDataCallLink: function(aUpdatedDataCall, aCurrentDataCall) {
|
|
// If network interface is changed, report as "deactivate".
|
|
if (aUpdatedDataCall.ifname != aCurrentDataCall.ifname) {
|
|
return "deactivate";
|
|
}
|
|
|
|
// If any existing address is missing, report as "deactivate".
|
|
for (let i = 0; i < aCurrentDataCall.addresses.length; i++) {
|
|
let address = aCurrentDataCall.addresses[i];
|
|
if (aUpdatedDataCall.addresses.indexOf(address) < 0) {
|
|
return "deactivate";
|
|
}
|
|
}
|
|
|
|
if (aCurrentDataCall.addresses.length != aUpdatedDataCall.addresses.length) {
|
|
// Since now all |aCurrentDataCall.addresses| are found in
|
|
// |aUpdatedDataCall.addresses|, this means one or more new addresses are
|
|
// reported.
|
|
return "changed";
|
|
}
|
|
|
|
let fields = ["gateways", "dnses"];
|
|
for (let i = 0; i < fields.length; i++) {
|
|
// Compare <datacall>.<field>.
|
|
let field = fields[i];
|
|
let lhs = aUpdatedDataCall[field], rhs = aCurrentDataCall[field];
|
|
if (lhs.length != rhs.length) {
|
|
return "changed";
|
|
}
|
|
for (let i = 0; i < lhs.length; i++) {
|
|
if (lhs[i] != rhs[i]) {
|
|
return "changed";
|
|
}
|
|
}
|
|
}
|
|
|
|
return "identical";
|
|
},
|
|
|
|
_getGeckoDataCallState:function (aDataCall) {
|
|
if (aDataCall.active == Ci.nsIDataCallInterface.DATACALL_STATE_ACTIVE_UP ||
|
|
aDataCall.active == Ci.nsIDataCallInterface.DATACALL_STATE_ACTIVE_DOWN) {
|
|
return NETWORK_STATE_CONNECTED;
|
|
}
|
|
|
|
return NETWORK_STATE_DISCONNECTED;
|
|
},
|
|
|
|
onSetupDataCallResult: function(aDataCall) {
|
|
this.debug("onSetupDataCallResult: " + JSON.stringify(aDataCall));
|
|
let errorMsg = aDataCall.errorMsg;
|
|
if (aDataCall.failCause &&
|
|
aDataCall.failCause != Ci.nsIDataCallInterface.DATACALL_FAIL_NONE) {
|
|
errorMsg =
|
|
RIL.RIL_DATACALL_FAILCAUSE_TO_GECKO_DATACALL_ERROR[aDataCall.failCause];
|
|
}
|
|
|
|
if (errorMsg) {
|
|
if (DEBUG) {
|
|
this.debug("SetupDataCall error for apn " + this.apnProfile.apn + ": " +
|
|
errorMsg + " (" + aDataCall.failCause + "), retry time: " +
|
|
aDataCall.suggestedRetryTime);
|
|
}
|
|
|
|
this.state = NETWORK_STATE_DISCONNECTED;
|
|
|
|
if (this.requestedNetworkIfaces.length === 0) {
|
|
if (DEBUG) this.debug("This DataCall is not requested anymore.");
|
|
return;
|
|
}
|
|
|
|
// Let DataCallHandler notify MobileConnectionService
|
|
this.dataCallHandler.notifyDataCallError(this);
|
|
|
|
// For suggestedRetryTime, the value of INT32_MAX(0x7fffffff) means no retry.
|
|
if (aDataCall.suggestedRetryTime === INT32_MAX ||
|
|
this.isPermanentFail(aDataCall.failCause, errorMsg)) {
|
|
if (DEBUG) this.debug("Data call error: no retry needed.");
|
|
return;
|
|
}
|
|
|
|
this.retry(aDataCall.suggestedRetryTime);
|
|
return;
|
|
}
|
|
|
|
this.apnRetryCounter = 0;
|
|
this.linkInfo.cid = aDataCall.cid;
|
|
|
|
if (this.requestedNetworkIfaces.length === 0) {
|
|
if (DEBUG) {
|
|
this.debug("State is connected, but no network interface requested" +
|
|
" this DataCall");
|
|
}
|
|
this.deactivate();
|
|
return;
|
|
}
|
|
|
|
this.linkInfo.ifname = aDataCall.ifname;
|
|
this.linkInfo.addresses = aDataCall.addresses ? aDataCall.addresses.split(" ") : [];
|
|
this.linkInfo.gateways = aDataCall.gateways ? aDataCall.gateways.split(" ") : [];
|
|
this.linkInfo.dnses = aDataCall.dnses ? aDataCall.dnses.split(" ") : [];
|
|
this.state = this._getGeckoDataCallState(aDataCall);
|
|
|
|
// Notify DataCallHandler about data call connected.
|
|
this.dataCallHandler.notifyDataCallChanged(this);
|
|
|
|
for (let i = 0; i < this.requestedNetworkIfaces.length; i++) {
|
|
this.requestedNetworkIfaces[i].notifyRILNetworkInterface();
|
|
}
|
|
},
|
|
|
|
onDeactivateDataCallResult: function() {
|
|
if (DEBUG) this.debug("onDeactivateDataCallResult");
|
|
|
|
this.reset();
|
|
|
|
if (this.requestedNetworkIfaces.length > 0) {
|
|
if (DEBUG) {
|
|
this.debug("State is disconnected/unknown, but this DataCall is" +
|
|
" requested.");
|
|
}
|
|
this.setup();
|
|
return;
|
|
}
|
|
|
|
// Notify DataCallHandler about data call disconnected.
|
|
this.dataCallHandler.notifyDataCallChanged(this);
|
|
},
|
|
|
|
onDataCallChanged: function(aUpdatedDataCall) {
|
|
if (DEBUG) {
|
|
this.debug("onDataCallChanged: " + JSON.stringify(aUpdatedDataCall));
|
|
}
|
|
|
|
if (this.state == NETWORK_STATE_CONNECTING ||
|
|
this.state == NETWORK_STATE_DISCONNECTING) {
|
|
if (DEBUG) {
|
|
this.debug("We are in connecting/disconnecting state, ignore any " +
|
|
"unsolicited event for now.");
|
|
}
|
|
return;
|
|
}
|
|
|
|
let dataCallState = this._getGeckoDataCallState(aUpdatedDataCall);
|
|
if (this.state == dataCallState &&
|
|
dataCallState != NETWORK_STATE_CONNECTED) {
|
|
return;
|
|
}
|
|
|
|
let newLinkInfo = {
|
|
ifname: aUpdatedDataCall.ifname,
|
|
addresses: aUpdatedDataCall.addresses ? aUpdatedDataCall.addresses.split(" ") : [],
|
|
dnses: aUpdatedDataCall.dnses ? aUpdatedDataCall.dnses.split(" ") : [],
|
|
gateways: aUpdatedDataCall.gateways ? aUpdatedDataCall.gateways.split(" ") : []
|
|
};
|
|
|
|
switch (dataCallState) {
|
|
case NETWORK_STATE_CONNECTED:
|
|
if (this.state == NETWORK_STATE_CONNECTED) {
|
|
let result =
|
|
this._compareDataCallLink(newLinkInfo, this.linkInfo);
|
|
|
|
if (result == "identical") {
|
|
if (DEBUG) this.debug("No changes in data call.");
|
|
return;
|
|
}
|
|
if (result == "deactivate") {
|
|
if (DEBUG) this.debug("Data link changed, cleanup.");
|
|
this.deactivate();
|
|
return;
|
|
}
|
|
// Minor change, just update and notify.
|
|
if (DEBUG) {
|
|
this.debug("Data link minor change, just update and notify.");
|
|
}
|
|
|
|
this.linkInfo.addresses = newLinkInfo.addresses.slice();
|
|
this.linkInfo.gateways = newLinkInfo.gateways.slice();
|
|
this.linkInfo.dnses = newLinkInfo.dnses.slice();
|
|
}
|
|
break;
|
|
case NETWORK_STATE_DISCONNECTED:
|
|
case NETWORK_STATE_UNKNOWN:
|
|
if (this.state == NETWORK_STATE_CONNECTED) {
|
|
// Notify first on unexpected data call disconnection.
|
|
this.state = dataCallState;
|
|
for (let i = 0; i < this.requestedNetworkIfaces.length; i++) {
|
|
this.requestedNetworkIfaces[i].notifyRILNetworkInterface();
|
|
}
|
|
}
|
|
this.reset();
|
|
|
|
if (this.requestedNetworkIfaces.length > 0) {
|
|
if (DEBUG) {
|
|
this.debug("State is disconnected/unknown, but this DataCall is" +
|
|
" requested.");
|
|
}
|
|
this.setup();
|
|
return;
|
|
}
|
|
break;
|
|
}
|
|
|
|
this.state = dataCallState;
|
|
|
|
// Notify DataCallHandler about data call changed.
|
|
this.dataCallHandler.notifyDataCallChanged(this);
|
|
|
|
for (let i = 0; i < this.requestedNetworkIfaces.length; i++) {
|
|
this.requestedNetworkIfaces[i].notifyRILNetworkInterface();
|
|
}
|
|
},
|
|
|
|
// Helpers
|
|
|
|
debug: function(aMsg) {
|
|
dump("-*- DataCall[" + this.clientId + ":" + this.apnProfile.apn + "]: " +
|
|
aMsg + "\n");
|
|
},
|
|
|
|
get connected() {
|
|
return this.state == NETWORK_STATE_CONNECTED;
|
|
},
|
|
|
|
isPermanentFail: function(aDataFailCause, aErrorMsg) {
|
|
// Check ril.h for 'no retry' data call fail causes.
|
|
if (aErrorMsg === RIL.GECKO_ERROR_RADIO_NOT_AVAILABLE ||
|
|
aErrorMsg === RIL.GECKO_ERROR_INVALID_PARAMETER ||
|
|
aDataFailCause === Ci.nsIDataCallInterface.DATACALL_FAIL_OPERATOR_BARRED ||
|
|
aDataFailCause === Ci.nsIDataCallInterface.DATACALL_FAIL_MISSING_UKNOWN_APN ||
|
|
aDataFailCause === Ci.nsIDataCallInterface.DATACALL_FAIL_UNKNOWN_PDP_ADDRESS_TYPE ||
|
|
aDataFailCause === Ci.nsIDataCallInterface.DATACALL_FAIL_USER_AUTHENTICATION ||
|
|
aDataFailCause === Ci.nsIDataCallInterface.DATACALL_FAIL_ACTIVATION_REJECT_GGSN ||
|
|
aDataFailCause === Ci.nsIDataCallInterface.DATACALL_FAIL_SERVICE_OPTION_NOT_SUPPORTED ||
|
|
aDataFailCause === Ci.nsIDataCallInterface.DATACALL_FAIL_SERVICE_OPTION_NOT_SUBSCRIBED ||
|
|
aDataFailCause === Ci.nsIDataCallInterface.DATACALL_FAIL_NSAPI_IN_USE ||
|
|
aDataFailCause === Ci.nsIDataCallInterface.DATACALL_FAIL_ONLY_IPV4_ALLOWED ||
|
|
aDataFailCause === Ci.nsIDataCallInterface.DATACALL_FAIL_ONLY_IPV6_ALLOWED ||
|
|
aDataFailCause === Ci.nsIDataCallInterface.DATACALL_FAIL_PROTOCOL_ERRORS ||
|
|
aDataFailCause === Ci.nsIDataCallInterface.DATACALL_FAIL_RADIO_POWER_OFF ||
|
|
aDataFailCause === Ci.nsIDataCallInterface.DATACALL_FAIL_TETHERED_CALL_ACTIVE) {
|
|
return true;
|
|
}
|
|
|
|
return false;
|
|
},
|
|
|
|
inRequestedTypes: function(aType) {
|
|
for (let i = 0; i < this.requestedNetworkIfaces.length; i++) {
|
|
if (this.requestedNetworkIfaces[i].type == aType) {
|
|
return true;
|
|
}
|
|
}
|
|
return false;
|
|
},
|
|
|
|
canHandleApn: function(aApnSetting) {
|
|
let isIdentical = this.apnProfile.apn == aApnSetting.apn &&
|
|
(this.apnProfile.user || '') == (aApnSetting.user || '') &&
|
|
(this.apnProfile.password || '') == (aApnSetting.password || '') &&
|
|
(this.apnProfile.authType || '') == (aApnSetting.authtype || '');
|
|
|
|
if (RILQUIRKS_HAVE_IPV6) {
|
|
isIdentical = isIdentical &&
|
|
(this.apnProfile.protocol || '') == (aApnSetting.protocol || '') &&
|
|
(this.apnProfile.roaming_protocol || '') == (aApnSetting.roaming_protocol || '');
|
|
}
|
|
|
|
return isIdentical;
|
|
},
|
|
|
|
resetLinkInfo: function() {
|
|
this.linkInfo.cid = null;
|
|
this.linkInfo.ifname = null;
|
|
this.linkInfo.addresses = [];
|
|
this.linkInfo.dnses = [];
|
|
this.linkInfo.gateways = [];
|
|
},
|
|
|
|
reset: function() {
|
|
this.resetLinkInfo();
|
|
|
|
this.state = NETWORK_STATE_UNKNOWN;
|
|
},
|
|
|
|
connect: function(aNetworkInterface) {
|
|
if (DEBUG) this.debug("connect: " + aNetworkInterface.type);
|
|
|
|
if (this.requestedNetworkIfaces.indexOf(aNetworkInterface) == -1) {
|
|
this.requestedNetworkIfaces.push(aNetworkInterface);
|
|
}
|
|
|
|
if (this.state == NETWORK_STATE_CONNECTING ||
|
|
this.state == NETWORK_STATE_DISCONNECTING) {
|
|
return;
|
|
}
|
|
if (this.state == NETWORK_STATE_CONNECTED) {
|
|
// This needs to run asynchronously, to behave the same way as the case of
|
|
// non-shared apn, see bug 1059110.
|
|
Services.tm.currentThread.dispatch(() => {
|
|
// Do not notify if state changed while this event was being dispatched,
|
|
// the state probably was notified already or need not to be notified.
|
|
if (aNetworkInterface.state == RIL.GECKO_NETWORK_STATE_CONNECTED) {
|
|
aNetworkInterface.notifyRILNetworkInterface();
|
|
}
|
|
}, Ci.nsIEventTarget.DISPATCH_NORMAL);
|
|
return;
|
|
}
|
|
|
|
// If retry mechanism is running on background, stop it since we are going
|
|
// to setup data call now.
|
|
if (this.timer) {
|
|
this.timer.cancel();
|
|
}
|
|
|
|
this.setup();
|
|
},
|
|
|
|
setup: function() {
|
|
if (DEBUG) {
|
|
this.debug("Going to set up data connection with APN " +
|
|
this.apnProfile.apn);
|
|
}
|
|
|
|
let connection =
|
|
gMobileConnectionService.getItemByServiceId(this.clientId);
|
|
let dataInfo = connection && connection.data;
|
|
if (dataInfo == null ||
|
|
dataInfo.state != RIL.GECKO_MOBILE_CONNECTION_STATE_REGISTERED ||
|
|
dataInfo.type == RIL.GECKO_MOBILE_CONNECTION_STATE_UNKNOWN) {
|
|
return;
|
|
}
|
|
|
|
let radioTechType = dataInfo.type;
|
|
let radioTechnology = RIL.GECKO_RADIO_TECH.indexOf(radioTechType);
|
|
let authType = RIL.RIL_DATACALL_AUTH_TO_GECKO.indexOf(this.apnProfile.authType);
|
|
// Use the default authType if the value in database is invalid.
|
|
// For the case that user might not select the authentication type.
|
|
if (authType == -1) {
|
|
if (DEBUG) {
|
|
this.debug("Invalid authType '" + this.apnProfile.authtype +
|
|
"', using '" + RIL.GECKO_DATACALL_AUTH_DEFAULT + "'");
|
|
}
|
|
authType = RIL.RIL_DATACALL_AUTH_TO_GECKO.indexOf(RIL.GECKO_DATACALL_AUTH_DEFAULT);
|
|
}
|
|
|
|
let pdpType = Ci.nsIDataCallInterface.DATACALL_PDP_TYPE_IPV4;
|
|
if (RILQUIRKS_HAVE_IPV6) {
|
|
pdpType = !dataInfo.roaming
|
|
? RIL.RIL_DATACALL_PDP_TYPES.indexOf(this.apnProfile.protocol)
|
|
: RIL.RIL_DATACALL_PDP_TYPES.indexOf(this.apnProfile.roaming_protocol);
|
|
if (pdpType == -1) {
|
|
if (DEBUG) {
|
|
this.debug("Invalid pdpType '" + (!dataInfo.roaming
|
|
? this.apnProfile.protocol
|
|
: this.apnProfile.roaming_protocol) +
|
|
"', using '" + RIL.GECKO_DATACALL_PDP_TYPE_DEFAULT + "'");
|
|
}
|
|
pdpType = RIL.RIL_DATACALL_PDP_TYPES.indexOf(RIL.GECKO_DATACALL_PDP_TYPE_DEFAULT);
|
|
}
|
|
}
|
|
|
|
let dcInterface = this.dataCallHandler.dataCallInterface;
|
|
dcInterface.setupDataCall(
|
|
this.apnProfile.apn, this.apnProfile.user, this.apnProfile.password,
|
|
authType, pdpType, {
|
|
QueryInterface: XPCOMUtils.generateQI([Ci.nsIDataCallCallback]),
|
|
notifySetupDataCallSuccess: (aDataCall) => {
|
|
this.onSetupDataCallResult(aDataCall);
|
|
},
|
|
notifyError: (aErrorMsg) => {
|
|
this.onSetupDataCallResult({errorMsg: aErrorMsg});
|
|
}
|
|
});
|
|
this.state = NETWORK_STATE_CONNECTING;
|
|
},
|
|
|
|
retry: function(aSuggestedRetryTime) {
|
|
let apnRetryTimer;
|
|
|
|
// We will retry the connection in increasing times
|
|
// based on the function: time = A * numer_of_retries^2 + B
|
|
if (this.apnRetryCounter >= this.NETWORK_APNRETRY_MAXRETRIES) {
|
|
this.apnRetryCounter = 0;
|
|
this.timer = null;
|
|
if (DEBUG) this.debug("Too many APN Connection retries - STOP retrying");
|
|
return;
|
|
}
|
|
|
|
// If there is a valid aSuggestedRetryTime, override the retry timer.
|
|
if (aSuggestedRetryTime !== undefined && aSuggestedRetryTime >= 0) {
|
|
apnRetryTimer = aSuggestedRetryTime / 1000;
|
|
} else {
|
|
apnRetryTimer = this.NETWORK_APNRETRY_FACTOR *
|
|
(this.apnRetryCounter * this.apnRetryCounter) +
|
|
this.NETWORK_APNRETRY_ORIGIN;
|
|
}
|
|
this.apnRetryCounter++;
|
|
if (DEBUG) {
|
|
this.debug("Data call - APN Connection Retry Timer (secs-counter): " +
|
|
apnRetryTimer + "-" + this.apnRetryCounter);
|
|
}
|
|
|
|
if (this.timer == null) {
|
|
// Event timer for connection retries
|
|
this.timer = Cc["@mozilla.org/timer;1"].createInstance(Ci.nsITimer);
|
|
}
|
|
this.timer.initWithCallback(this, apnRetryTimer * 1000,
|
|
Ci.nsITimer.TYPE_ONE_SHOT);
|
|
},
|
|
|
|
disconnect: function(aNetworkInterface) {
|
|
if (DEBUG) this.debug("disconnect: " + aNetworkInterface.type);
|
|
|
|
let index = this.requestedNetworkIfaces.indexOf(aNetworkInterface);
|
|
if (index != -1) {
|
|
this.requestedNetworkIfaces.splice(index, 1);
|
|
|
|
if (this.state == NETWORK_STATE_DISCONNECTED ||
|
|
this.state == NETWORK_STATE_UNKNOWN) {
|
|
if (this.timer) {
|
|
this.timer.cancel();
|
|
}
|
|
this.reset();
|
|
return;
|
|
}
|
|
|
|
// Notify the DISCONNECTED event immediately after network interface is
|
|
// removed from requestedNetworkIfaces, to make the DataCall, shared or
|
|
// not, to have the same behavior.
|
|
Services.tm.currentThread.dispatch(() => {
|
|
// Do not notify if state changed while this event was being dispatched,
|
|
// the state probably was notified already or need not to be notified.
|
|
if (aNetworkInterface.state == RIL.GECKO_NETWORK_STATE_DISCONNECTED) {
|
|
aNetworkInterface.notifyRILNetworkInterface();
|
|
|
|
// Clear link info after notifying NetworkManager.
|
|
if (this.requestedNetworkIfaces.length === 0) {
|
|
this.resetLinkInfo();
|
|
}
|
|
}
|
|
}, Ci.nsIEventTarget.DISPATCH_NORMAL);
|
|
}
|
|
|
|
// Only deactivate data call if no more network interface needs this
|
|
// DataCall and if state is CONNECTED, for other states, we simply remove
|
|
// the network interface from requestedNetworkIfaces.
|
|
if (this.requestedNetworkIfaces.length > 0 ||
|
|
this.state != NETWORK_STATE_CONNECTED) {
|
|
return;
|
|
}
|
|
|
|
this.deactivate();
|
|
},
|
|
|
|
deactivate: function() {
|
|
let reason = Ci.nsINetworkInterface.DATACALL_DEACTIVATE_NO_REASON;
|
|
if (DEBUG) {
|
|
this.debug("Going to disconnect data connection cid " + this.linkInfo.cid);
|
|
}
|
|
|
|
let dcInterface = this.dataCallHandler.dataCallInterface;
|
|
dcInterface.deactivateDataCall(this.linkInfo.cid, reason, {
|
|
QueryInterface: XPCOMUtils.generateQI([Ci.nsIDataCallCallback]),
|
|
notifySuccess: () => {
|
|
this.onDeactivateDataCallResult();
|
|
},
|
|
notifyError: (aErrorMsg) => {
|
|
this.onDeactivateDataCallResult();
|
|
}
|
|
});
|
|
|
|
this.state = NETWORK_STATE_DISCONNECTING;
|
|
},
|
|
|
|
// Entry method for timer events. Used to reconnect to a failed APN
|
|
notify: function(aTimer) {
|
|
this.setup();
|
|
},
|
|
|
|
shutdown: function() {
|
|
if (this.timer) {
|
|
this.timer.cancel();
|
|
this.timer = null;
|
|
}
|
|
}
|
|
};
|
|
|
|
function RILNetworkInterface(aDataCallHandler, aType, aApnSetting, aDataCall) {
|
|
if (!aDataCall) {
|
|
throw new Error("No dataCall for RILNetworkInterface: " + type);
|
|
}
|
|
|
|
this.dataCallHandler = aDataCallHandler;
|
|
this.type = aType;
|
|
this.apnSetting = aApnSetting;
|
|
this.dataCall = aDataCall;
|
|
|
|
this.enabled = false;
|
|
}
|
|
|
|
RILNetworkInterface.prototype = {
|
|
classID: RILNETWORKINTERFACE_CID,
|
|
classInfo: XPCOMUtils.generateCI({classID: RILNETWORKINTERFACE_CID,
|
|
classDescription: "RILNetworkInterface",
|
|
interfaces: [Ci.nsINetworkInterface,
|
|
Ci.nsIRilNetworkInterface]}),
|
|
QueryInterface: XPCOMUtils.generateQI([Ci.nsINetworkInterface,
|
|
Ci.nsIRilNetworkInterface]),
|
|
|
|
// Hold reference to DataCall object which is determined at initilization.
|
|
dataCall: null,
|
|
|
|
// If this RILNetworkInterface type is enabled or not.
|
|
enabled: null,
|
|
|
|
/**
|
|
* nsINetworkInterface Implementation
|
|
*/
|
|
|
|
get state() {
|
|
if (!this.dataCall.inRequestedTypes(this.type)) {
|
|
return NETWORK_STATE_DISCONNECTED;
|
|
}
|
|
return this.dataCall.state;
|
|
},
|
|
|
|
type: null,
|
|
|
|
get name() {
|
|
return this.dataCall.linkInfo.ifname;
|
|
},
|
|
|
|
get httpProxyHost() {
|
|
return this.apnSetting.proxy || "";
|
|
},
|
|
|
|
get httpProxyPort() {
|
|
return this.apnSetting.port || "";
|
|
},
|
|
|
|
getAddresses: function(aIps, aPrefixLengths) {
|
|
let addresses = this.dataCall.linkInfo.addresses;
|
|
|
|
let ips = [];
|
|
let prefixLengths = [];
|
|
for (let i = 0; i < addresses.length; i++) {
|
|
let [ip, prefixLength] = addresses[i].split("/");
|
|
ips.push(ip);
|
|
prefixLengths.push(prefixLength);
|
|
}
|
|
|
|
aIps.value = ips.slice();
|
|
aPrefixLengths.value = prefixLengths.slice();
|
|
|
|
return ips.length;
|
|
},
|
|
|
|
getGateways: function(aCount) {
|
|
let linkInfo = this.dataCall.linkInfo;
|
|
|
|
if (aCount) {
|
|
aCount.value = linkInfo.gateways.length;
|
|
}
|
|
return linkInfo.gateways.slice();
|
|
},
|
|
|
|
getDnses: function(aCount) {
|
|
let linkInfo = this.dataCall.linkInfo;
|
|
|
|
if (aCount) {
|
|
aCount.value = linkInfo.dnses.length;
|
|
}
|
|
return linkInfo.dnses.slice();
|
|
},
|
|
|
|
/**
|
|
* nsIRilNetworkInterface Implementation
|
|
*/
|
|
|
|
get serviceId() {
|
|
return this.dataCallHandler.clientId;
|
|
},
|
|
|
|
get iccId() {
|
|
let icc = gIccService.getIccByServiceId(this.dataCallHandler.clientId);
|
|
let iccInfo = icc && icc.iccInfo;
|
|
|
|
return iccInfo && iccInfo.iccid;
|
|
},
|
|
|
|
get mmsc() {
|
|
if (this.type != NETWORK_TYPE_MOBILE_MMS) {
|
|
if (DEBUG) this.debug("Error! Only MMS network can get MMSC.");
|
|
throw Cr.NS_ERROR_UNEXPECTED;
|
|
}
|
|
|
|
return this.apnSetting.mmsc || "";
|
|
},
|
|
|
|
get mmsProxy() {
|
|
if (this.type != NETWORK_TYPE_MOBILE_MMS) {
|
|
if (DEBUG) this.debug("Error! Only MMS network can get MMS proxy.");
|
|
throw Cr.NS_ERROR_UNEXPECTED;
|
|
}
|
|
|
|
return this.apnSetting.mmsproxy || "";
|
|
},
|
|
|
|
get mmsPort() {
|
|
if (this.type != NETWORK_TYPE_MOBILE_MMS) {
|
|
if (DEBUG) this.debug("Error! Only MMS network can get MMS port.");
|
|
throw Cr.NS_ERROR_UNEXPECTED;
|
|
}
|
|
|
|
// Note: Port 0 is reserved, so we treat it as invalid as well.
|
|
// See http://www.iana.org/assignments/port-numbers
|
|
return this.apnSetting.mmsport || -1;
|
|
},
|
|
|
|
// Helpers
|
|
|
|
debug: function(aMsg) {
|
|
dump("-*- RILNetworkInterface[" + this.dataCallHandler.clientId + ":" +
|
|
this.type + "]: " + aMsg + "\n");
|
|
},
|
|
|
|
apnSetting: null,
|
|
|
|
get connected() {
|
|
return this.state == NETWORK_STATE_CONNECTED;
|
|
},
|
|
|
|
notifyRILNetworkInterface: function() {
|
|
if (DEBUG) {
|
|
this.debug("notifyRILNetworkInterface type: " + this.type + ", state: " +
|
|
this.state);
|
|
}
|
|
|
|
gNetworkManager.updateNetworkInterface(this);
|
|
},
|
|
|
|
connect: function() {
|
|
this.enabled = true;
|
|
|
|
this.dataCall.connect(this);
|
|
},
|
|
|
|
disconnect: function() {
|
|
if (!this.enabled) {
|
|
return;
|
|
}
|
|
this.enabled = false;
|
|
|
|
this.dataCall.disconnect(this);
|
|
},
|
|
|
|
shutdown: function() {
|
|
this.dataCall.shutdown();
|
|
this.dataCall = null;
|
|
}
|
|
};
|
|
|
|
this.NSGetFactory = XPCOMUtils.generateNSGetFactory([DataCallManager]); |