зеркало из https://github.com/mozilla/gecko-dev.git
373 строки
12 KiB
JavaScript
373 строки
12 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/. */
|
|
|
|
this.EXPORTED_SYMBOLS = ["UpdateUtils"];
|
|
|
|
const { classes: Cc, interfaces: Ci, utils: Cu } = Components;
|
|
|
|
Cu.import("resource://gre/modules/AppConstants.jsm");
|
|
Cu.import("resource://gre/modules/Services.jsm");
|
|
Cu.import("resource://gre/modules/XPCOMUtils.jsm");
|
|
Cu.import("resource://gre/modules/NetUtil.jsm");
|
|
Cu.import("resource://gre/modules/Preferences.jsm");
|
|
Cu.import("resource://gre/modules/ctypes.jsm");
|
|
|
|
const FILE_UPDATE_LOCALE = "update.locale";
|
|
const PREF_APP_DISTRIBUTION = "distribution.id";
|
|
const PREF_APP_DISTRIBUTION_VERSION = "distribution.version";
|
|
const PREF_APP_B2G_VERSION = "b2g.version";
|
|
const PREF_APP_UPDATE_CUSTOM = "app.update.custom";
|
|
const PREF_APP_UPDATE_IMEI_HASH = "app.update.imei_hash";
|
|
|
|
|
|
this.UpdateUtils = {
|
|
/**
|
|
* Read the update channel from defaults only. We do this to ensure that
|
|
* the channel is tightly coupled with the application and does not apply
|
|
* to other instances of the application that may use the same profile.
|
|
*
|
|
* @param [optional] aIncludePartners
|
|
* Whether or not to include the partner bits. Default: true.
|
|
*/
|
|
getUpdateChannel(aIncludePartners = true) {
|
|
let channel = AppConstants.MOZ_UPDATE_CHANNEL;
|
|
let defaults = Services.prefs.getDefaultBranch(null);
|
|
try {
|
|
channel = defaults.getCharPref("app.update.channel");
|
|
} catch (e) {
|
|
// use default value when pref not found
|
|
}
|
|
|
|
if (aIncludePartners) {
|
|
try {
|
|
let partners = Services.prefs.getChildList("app.partner.").sort();
|
|
if (partners.length) {
|
|
channel += "-cck";
|
|
partners.forEach(function(prefName) {
|
|
channel += "-" + Services.prefs.getCharPref(prefName);
|
|
});
|
|
}
|
|
} catch (e) {
|
|
Cu.reportError(e);
|
|
}
|
|
}
|
|
|
|
return channel;
|
|
},
|
|
|
|
get UpdateChannel() {
|
|
return this.getUpdateChannel();
|
|
},
|
|
|
|
/**
|
|
* Formats a URL by replacing %...% values with OS, build and locale specific
|
|
* values.
|
|
*
|
|
* @param url
|
|
* The URL to format.
|
|
* @return The formatted URL.
|
|
*/
|
|
formatUpdateURL(url) {
|
|
url = url.replace(/%PRODUCT%/g, Services.appinfo.name);
|
|
url = url.replace(/%VERSION%/g, Services.appinfo.version);
|
|
url = url.replace(/%BUILD_ID%/g, Services.appinfo.appBuildID);
|
|
url = url.replace(/%BUILD_TARGET%/g, Services.appinfo.OS + "_" + this.ABI);
|
|
url = url.replace(/%OS_VERSION%/g, this.OSVersion);
|
|
url = url.replace(/%SYSTEM_CAPABILITIES%/g, gSystemCapabilities);
|
|
if (/%LOCALE%/.test(url)) {
|
|
url = url.replace(/%LOCALE%/g, this.Locale);
|
|
}
|
|
url = url.replace(/%CHANNEL%/g, this.UpdateChannel);
|
|
url = url.replace(/%PLATFORM_VERSION%/g, Services.appinfo.platformVersion);
|
|
url = url.replace(/%DISTRIBUTION%/g,
|
|
getDistributionPrefValue(PREF_APP_DISTRIBUTION));
|
|
url = url.replace(/%DISTRIBUTION_VERSION%/g,
|
|
getDistributionPrefValue(PREF_APP_DISTRIBUTION_VERSION));
|
|
url = url.replace(/%CUSTOM%/g, Preferences.get(PREF_APP_UPDATE_CUSTOM, ""));
|
|
url = url.replace(/\+/g, "%2B");
|
|
|
|
if (AppConstants.platform == "gonk") {
|
|
let sysLibs = {};
|
|
Cu.import("resource://gre/modules/systemlibs.js", sysLibs);
|
|
let productDevice = sysLibs.libcutils.property_get("ro.product.device");
|
|
let buildType = sysLibs.libcutils.property_get("ro.build.type");
|
|
url = url.replace(/%PRODUCT_MODEL%/g,
|
|
sysLibs.libcutils.property_get("ro.product.model"));
|
|
if (buildType == "user" || buildType == "userdebug") {
|
|
url = url.replace(/%PRODUCT_DEVICE%/g, productDevice);
|
|
} else {
|
|
url = url.replace(/%PRODUCT_DEVICE%/g, productDevice + "-" + buildType);
|
|
}
|
|
url = url.replace(/%B2G_VERSION%/g,
|
|
Preferences.get(PREF_APP_B2G_VERSION, null));
|
|
url = url.replace(/%IMEI%/g,
|
|
Preferences.get(PREF_APP_UPDATE_IMEI_HASH, "default"));
|
|
}
|
|
|
|
return url;
|
|
}
|
|
};
|
|
|
|
/* Get the distribution pref values, from defaults only */
|
|
function getDistributionPrefValue(aPrefName) {
|
|
var prefValue = "default";
|
|
|
|
try {
|
|
prefValue = Services.prefs.getDefaultBranch(null).getCharPref(aPrefName);
|
|
} catch (e) {
|
|
// use default when pref not found
|
|
}
|
|
|
|
return prefValue;
|
|
}
|
|
|
|
/**
|
|
* Gets the locale from the update.locale file for replacing %LOCALE% in the
|
|
* update url. The update.locale file can be located in the application
|
|
* directory or the GRE directory with preference given to it being located in
|
|
* the application directory.
|
|
*/
|
|
XPCOMUtils.defineLazyGetter(UpdateUtils, "Locale", function() {
|
|
let channel;
|
|
let locale;
|
|
for (let res of ["app", "gre"]) {
|
|
channel = NetUtil.newChannel({
|
|
uri: "resource://" + res + "/" + FILE_UPDATE_LOCALE,
|
|
contentPolicyType: Ci.nsIContentPolicy.TYPE_INTERNAL_XMLHTTPREQUEST,
|
|
loadUsingSystemPrincipal: true
|
|
});
|
|
try {
|
|
let inputStream = channel.open2();
|
|
locale = NetUtil.readInputStreamToString(inputStream, inputStream.available());
|
|
} catch (e) {}
|
|
if (locale)
|
|
return locale.trim();
|
|
}
|
|
|
|
Cu.reportError(FILE_UPDATE_LOCALE + " file doesn't exist in either the " +
|
|
"application or GRE directories");
|
|
|
|
return null;
|
|
});
|
|
|
|
/**
|
|
* Provides adhoc system capability information for application update.
|
|
*/
|
|
XPCOMUtils.defineLazyGetter(this, "gSystemCapabilities", function aus_gSC() {
|
|
if (AppConstants.platform == "win") {
|
|
const PF_MMX_INSTRUCTIONS_AVAILABLE = 3; // MMX
|
|
const PF_XMMI_INSTRUCTIONS_AVAILABLE = 6; // SSE
|
|
const PF_XMMI64_INSTRUCTIONS_AVAILABLE = 10; // SSE2
|
|
const PF_SSE3_INSTRUCTIONS_AVAILABLE = 13; // SSE3
|
|
|
|
let lib = ctypes.open("kernel32.dll");
|
|
let IsProcessorFeaturePresent = lib.declare("IsProcessorFeaturePresent",
|
|
ctypes.winapi_abi,
|
|
ctypes.int32_t, /* success */
|
|
ctypes.uint32_t); /* DWORD */
|
|
let instructionSet = "unknown";
|
|
try {
|
|
if (IsProcessorFeaturePresent(PF_SSE3_INSTRUCTIONS_AVAILABLE)) {
|
|
instructionSet = "SSE3";
|
|
} else if (IsProcessorFeaturePresent(PF_XMMI64_INSTRUCTIONS_AVAILABLE)) {
|
|
instructionSet = "SSE2";
|
|
} else if (IsProcessorFeaturePresent(PF_XMMI_INSTRUCTIONS_AVAILABLE)) {
|
|
instructionSet = "SSE";
|
|
} else if (IsProcessorFeaturePresent(PF_MMX_INSTRUCTIONS_AVAILABLE)) {
|
|
instructionSet = "MMX";
|
|
}
|
|
} catch (e) {
|
|
instructionSet = "error";
|
|
Cu.reportError("Error getting processor instruction set. " +
|
|
"Exception: " + e);
|
|
}
|
|
|
|
lib.close();
|
|
return instructionSet;
|
|
}
|
|
|
|
if (AppConstants == "linux") {
|
|
let instructionSet = "unknown";
|
|
if (navigator.cpuHasSSE2) {
|
|
instructionSet = "SSE2";
|
|
}
|
|
return instructionSet;
|
|
}
|
|
|
|
return "NA"
|
|
});
|
|
|
|
/* Windows only getter that returns the processor architecture. */
|
|
XPCOMUtils.defineLazyGetter(this, "gWinCPUArch", function aus_gWinCPUArch() {
|
|
// Get processor architecture
|
|
let arch = "unknown";
|
|
|
|
const WORD = ctypes.uint16_t;
|
|
const DWORD = ctypes.uint32_t;
|
|
|
|
// This structure is described at:
|
|
// http://msdn.microsoft.com/en-us/library/ms724958%28v=vs.85%29.aspx
|
|
const SYSTEM_INFO = new ctypes.StructType("SYSTEM_INFO",
|
|
[
|
|
{wProcessorArchitecture: WORD},
|
|
{wReserved: WORD},
|
|
{dwPageSize: DWORD},
|
|
{lpMinimumApplicationAddress: ctypes.voidptr_t},
|
|
{lpMaximumApplicationAddress: ctypes.voidptr_t},
|
|
{dwActiveProcessorMask: DWORD.ptr},
|
|
{dwNumberOfProcessors: DWORD},
|
|
{dwProcessorType: DWORD},
|
|
{dwAllocationGranularity: DWORD},
|
|
{wProcessorLevel: WORD},
|
|
{wProcessorRevision: WORD}
|
|
]);
|
|
|
|
let kernel32 = false;
|
|
try {
|
|
kernel32 = ctypes.open("Kernel32");
|
|
} catch (e) {
|
|
Cu.reportError("Unable to open kernel32! Exception: " + e);
|
|
}
|
|
|
|
if (kernel32) {
|
|
try {
|
|
let GetNativeSystemInfo = kernel32.declare("GetNativeSystemInfo",
|
|
ctypes.default_abi,
|
|
ctypes.void_t,
|
|
SYSTEM_INFO.ptr);
|
|
let winSystemInfo = SYSTEM_INFO();
|
|
// Default to unknown
|
|
winSystemInfo.wProcessorArchitecture = 0xffff;
|
|
|
|
GetNativeSystemInfo(winSystemInfo.address());
|
|
switch (winSystemInfo.wProcessorArchitecture) {
|
|
case 9:
|
|
arch = "x64";
|
|
break;
|
|
case 6:
|
|
arch = "IA64";
|
|
break;
|
|
case 0:
|
|
arch = "x86";
|
|
break;
|
|
}
|
|
} catch (e) {
|
|
Cu.reportError("Error getting processor architecture. " +
|
|
"Exception: " + e);
|
|
} finally {
|
|
kernel32.close();
|
|
}
|
|
}
|
|
|
|
return arch;
|
|
});
|
|
|
|
XPCOMUtils.defineLazyGetter(UpdateUtils, "ABI", function() {
|
|
let abi = null;
|
|
try {
|
|
abi = Services.appinfo.XPCOMABI;
|
|
} catch (e) {
|
|
Cu.reportError("XPCOM ABI unknown");
|
|
}
|
|
|
|
if (AppConstants.platform == "macosx") {
|
|
// Mac universal build should report a different ABI than either macppc
|
|
// or mactel.
|
|
let macutils = Cc["@mozilla.org/xpcom/mac-utils;1"].
|
|
getService(Ci.nsIMacUtils);
|
|
|
|
if (macutils.isUniversalBinary) {
|
|
abi += "-u-" + macutils.architecturesInBinary;
|
|
}
|
|
} else if (AppConstants.platform == "win") {
|
|
// Windows build should report the CPU architecture that it's running on.
|
|
abi += "-" + gWinCPUArch;
|
|
}
|
|
return abi;
|
|
});
|
|
|
|
XPCOMUtils.defineLazyGetter(UpdateUtils, "OSVersion", function() {
|
|
let osVersion;
|
|
try {
|
|
osVersion = Services.sysinfo.getProperty("name") + " " +
|
|
Services.sysinfo.getProperty("version");
|
|
} catch (e) {
|
|
Cu.reportError("OS Version unknown.");
|
|
}
|
|
|
|
if (osVersion) {
|
|
if (AppConstants.platform == "win") {
|
|
const BYTE = ctypes.uint8_t;
|
|
const WORD = ctypes.uint16_t;
|
|
const DWORD = ctypes.uint32_t;
|
|
const WCHAR = ctypes.char16_t;
|
|
const BOOL = ctypes.int;
|
|
|
|
// This structure is described at:
|
|
// http://msdn.microsoft.com/en-us/library/ms724833%28v=vs.85%29.aspx
|
|
const SZCSDVERSIONLENGTH = 128;
|
|
const OSVERSIONINFOEXW = new ctypes.StructType("OSVERSIONINFOEXW",
|
|
[
|
|
{dwOSVersionInfoSize: DWORD},
|
|
{dwMajorVersion: DWORD},
|
|
{dwMinorVersion: DWORD},
|
|
{dwBuildNumber: DWORD},
|
|
{dwPlatformId: DWORD},
|
|
{szCSDVersion: ctypes.ArrayType(WCHAR, SZCSDVERSIONLENGTH)},
|
|
{wServicePackMajor: WORD},
|
|
{wServicePackMinor: WORD},
|
|
{wSuiteMask: WORD},
|
|
{wProductType: BYTE},
|
|
{wReserved: BYTE}
|
|
]);
|
|
|
|
let kernel32 = false;
|
|
try {
|
|
kernel32 = ctypes.open("Kernel32");
|
|
} catch (e) {
|
|
Cu.reportError("Unable to open kernel32! " + e);
|
|
osVersion += ".unknown (unknown)";
|
|
}
|
|
|
|
if (kernel32) {
|
|
try {
|
|
// Get Service pack info
|
|
try {
|
|
let GetVersionEx = kernel32.declare("GetVersionExW",
|
|
ctypes.default_abi,
|
|
BOOL,
|
|
OSVERSIONINFOEXW.ptr);
|
|
let winVer = OSVERSIONINFOEXW();
|
|
winVer.dwOSVersionInfoSize = OSVERSIONINFOEXW.size;
|
|
|
|
if (0 !== GetVersionEx(winVer.address())) {
|
|
osVersion += "." + winVer.wServicePackMajor +
|
|
"." + winVer.wServicePackMinor;
|
|
} else {
|
|
Cu.reportError("Unknown failure in GetVersionEX (returned 0)");
|
|
osVersion += ".unknown";
|
|
}
|
|
} catch (e) {
|
|
Cu.reportError("Error getting service pack information. Exception: " + e);
|
|
osVersion += ".unknown";
|
|
}
|
|
} finally {
|
|
kernel32.close();
|
|
}
|
|
|
|
// Add processor architecture
|
|
osVersion += " (" + gWinCPUArch + ")";
|
|
}
|
|
}
|
|
|
|
try {
|
|
osVersion += " (" + Services.sysinfo.getProperty("secondaryLibrary") + ")";
|
|
} catch (e) {
|
|
// Not all platforms have a secondary widget library, so an error is nothing to worry about.
|
|
}
|
|
osVersion = encodeURIComponent(osVersion);
|
|
}
|
|
return osVersion;
|
|
});
|