/* 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/. */ var EXPORTED_SYMBOLS = ["UpdateUtils"]; ChromeUtils.import("resource://gre/modules/AppConstants.jsm"); ChromeUtils.import("resource://gre/modules/Services.jsm"); ChromeUtils.import("resource://gre/modules/XPCOMUtils.jsm"); ChromeUtils.import("resource://gre/modules/ctypes.jsm"); XPCOMUtils.defineLazyGlobalGetters(this, ["fetch"]); /* globals fetch */ ChromeUtils.defineModuleGetter(this, "WindowsRegistry", "resource://gre/modules/WindowsRegistry.jsm"); const FILE_UPDATE_LOCALE = "update.locale"; const PREF_APP_DISTRIBUTION = "distribution.id"; const PREF_APP_DISTRIBUTION_VERSION = "distribution.version"; const PREF_APP_UPDATE_CUSTOM = "app.update.custom"; var UpdateUtils = { _locale: undefined, /** * 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 defaults = Services.prefs.getDefaultBranch(null); let channel = defaults.getCharPref("app.update.channel", AppConstants.MOZ_UPDATE_CHANNEL); 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. */ async formatUpdateURL(url) { const locale = await this.getLocale(); return url.replace(/%(\w+)%/g, (match, name) => { switch (name) { case "PRODUCT": return Services.appinfo.name; case "VERSION": return Services.appinfo.version; case "BUILD_ID": return Services.appinfo.appBuildID; case "BUILD_TARGET": return Services.appinfo.OS + "_" + this.ABI; case "OS_VERSION": return this.OSVersion; case "LOCALE": return locale; case "CHANNEL": return this.UpdateChannel; case "PLATFORM_VERSION": return Services.appinfo.platformVersion; case "SYSTEM_CAPABILITIES": return getSystemCapabilities(); case "CUSTOM": return Services.prefs.getStringPref(PREF_APP_UPDATE_CUSTOM, ""); case "DISTRIBUTION": return getDistributionPrefValue(PREF_APP_DISTRIBUTION); case "DISTRIBUTION_VERSION": return getDistributionPrefValue(PREF_APP_DISTRIBUTION_VERSION); } return match; }).replace(/\+/g, "%2B"); }, /** * 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. */ async getLocale() { if (this._locale !== undefined) { return this._locale; } for (let res of ["app", "gre"]) { const url = "resource://" + res + "/" + FILE_UPDATE_LOCALE; let data; try { data = await fetch(url); } catch (e) { continue; } const locale = await data.text(); if (locale) { return this._locale = locale.trim(); } } Cu.reportError(FILE_UPDATE_LOCALE + " file doesn't exist in either the " + "application or GRE directories"); return this._locale = null; } }; /* Get the distribution pref values, from defaults only */ function getDistributionPrefValue(aPrefName) { return Services.prefs.getDefaultBranch(null).getCharPref(aPrefName, "default"); } function getSystemCapabilities() { return "ISET:" + gInstructionSet + ",MEM:" + getMemoryMB(); } /** * Gets the RAM size in megabytes. This will round the value because sysinfo * doesn't always provide RAM in multiples of 1024. */ function getMemoryMB() { let memoryMB = "unknown"; try { memoryMB = Services.sysinfo.getProperty("memsize"); if (memoryMB) { memoryMB = Math.round(memoryMB / 1024 / 1024); } } catch (e) { Cu.reportError("Error getting system info memsize property. " + "Exception: " + e); } return memoryMB; } /** * Gets the supported CPU instruction set. */ XPCOMUtils.defineLazyGetter(this, "gInstructionSet", function aus_gIS() { const CPU_EXTENSIONS = ["hasSSE4_2", "hasSSE4_1", "hasSSE4A", "hasSSSE3", "hasSSE3", "hasSSE2", "hasSSE", "hasMMX", "hasNEON", "hasARMv7", "hasARMv6"]; for (let ext of CPU_EXTENSIONS) { if (Services.sysinfo.getProperty(ext)) { return ext.substring(3); } } return "unknown"; }); /* 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.winapi_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 == "win") { // Windows build should report the CPU architecture that it's running on. abi += "-" + gWinCPUArch; } if (AppConstants.ASAN) { // Allow ASan builds to receive their own updates abi += "-asan"; } 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.winapi_abi, BOOL, OSVERSIONINFOEXW.ptr); let winVer = OSVERSIONINFOEXW(); winVer.dwOSVersionInfoSize = OSVERSIONINFOEXW.size; if (0 !== GetVersionEx(winVer.address())) { osVersion += "." + winVer.wServicePackMajor + "." + winVer.wServicePackMinor + "." + winVer.dwBuildNumber; } else { Cu.reportError("Unknown failure in GetVersionEX (returned 0)"); osVersion += ".unknown"; } } catch (e) { Cu.reportError("Error getting service pack information. Exception: " + e); osVersion += ".unknown"; } if (Services.vc.compare(Services.sysinfo.getProperty("version"), "10") >= 0) { const WINDOWS_UBR_KEY_PATH = "SOFTWARE\\Microsoft\\Windows NT\\CurrentVersion"; let ubr = WindowsRegistry.readRegKey(Ci.nsIWindowsRegKey.ROOT_KEY_LOCAL_MACHINE, WINDOWS_UBR_KEY_PATH, "UBR", Ci.nsIWindowsRegKey.WOW64_64); osVersion += (ubr !== undefined) ? "." + ubr : ".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; });