Received: by 10.213.65.68 with SMTP id h4csp589606imn; Tue, 13 Mar 2018 14:07:50 -0700 (PDT) X-Google-Smtp-Source: AG47ELtReEl2a4fqMTcDT6ak3UfztmG6vrgoBTIBooQp63wL5/Z64bN+RE/CzJ3HXMMF1tb46P8M X-Received: by 10.99.102.132 with SMTP id a126mr350609pgc.385.1520975270633; Tue, 13 Mar 2018 14:07:50 -0700 (PDT) ARC-Seal: i=1; a=rsa-sha256; t=1520975270; cv=none; d=google.com; s=arc-20160816; b=G/X+1wqJg15OPtqok0D/FWI42YyKBWHkDUIWnBvGh90PRnWaCR498SLPL2MrlieZ8G 2r51y+B8m1T7YNL+HvdRGfqSSkPbCUmxde9fTbQBF015fA+rXshN5kQJmHyZNEVTy5Lf bBow3b25jyiLKEnOBpm6ikaVD73lezf6SkuMeXE7RcQ7WIm1FEQG/WZEZ9dtnNPfYxg5 a9OVxMDw+7XtE1ENXevdQQ356rLeYLh1htAN/zREa8KKdwymCmi+m2wP7lVD91ELVL75 McOvuWIS5Qqpyip7+V+xrll9Lnny8sKj1RI61k+mnxjh7Ew0TnzilBz42KLD+xbszH9i a2Vg== ARC-Message-Signature: i=1; a=rsa-sha256; c=relaxed/relaxed; d=google.com; s=arc-20160816; h=list-id:precedence:sender:content-transfer-encoding :content-language:in-reply-to:mime-version:user-agent:date :message-id:references:cc:to:subject:from:arc-authentication-results; bh=+TejmNLK3d4Ot2YAFhOcWX8DNaKy5I2wo/pldABs8ek=; b=s12Sf+hlVCl+8JbFPVtJpTVH5oRf4X6tQNhFGuOVexi59UU+DpIN+KKHDKIPrmAzyy NTRM82stihqza1lHZRVjujK2hod/8lRa9yk6iw7w32MMmckapRX7i5yG6IhcmMAGr//P FiE9R8pr33H5DbJP94cHPolQBZHIAyuJ5QWAQS2xPeBY38vC2NLDHW8AO5e8PRIVjS31 oDgM2YqS55wpWpraVejjRblfgjK4AkXAUfh/BxxPbFA4+bfddXN8A+7SqUbJMyv73TF2 4M2ZSNXP4cFoaAgcV3kKmlLLvJAH/ln/7sMBQYidYOngBSdDczfDwdgs9l0yCtV2wRwG jtEg== ARC-Authentication-Results: i=1; mx.google.com; spf=pass (google.com: best guess record for domain of linux-kernel-owner@vger.kernel.org designates 209.132.180.67 as permitted sender) smtp.mailfrom=linux-kernel-owner@vger.kernel.org Return-Path: Received: from vger.kernel.org (vger.kernel.org. [209.132.180.67]) by mx.google.com with ESMTP id g1si659823pgq.219.2018.03.13.14.07.36; Tue, 13 Mar 2018 14:07:50 -0700 (PDT) Received-SPF: pass (google.com: best guess record for domain of linux-kernel-owner@vger.kernel.org designates 209.132.180.67 as permitted sender) client-ip=209.132.180.67; Authentication-Results: mx.google.com; spf=pass (google.com: best guess record for domain of linux-kernel-owner@vger.kernel.org designates 209.132.180.67 as permitted sender) smtp.mailfrom=linux-kernel-owner@vger.kernel.org Received: (majordomo@vger.kernel.org) by vger.kernel.org via listexpand id S1753141AbeCMVG1 (ORCPT + 99 others); Tue, 13 Mar 2018 17:06:27 -0400 Received: from vps-vb.mhejs.net ([37.28.154.113]:45744 "EHLO vps-vb.mhejs.net" rhost-flags-OK-OK-OK-OK) by vger.kernel.org with ESMTP id S1751903AbeCMVGZ (ORCPT ); Tue, 13 Mar 2018 17:06:25 -0400 Received: by vps-vb.mhejs.net with esmtps (TLSv1.2:ECDHE-RSA-AES128-GCM-SHA256:128) (Exim 4.90_1) (envelope-from ) id 1evr7n-0004S2-Gq; Tue, 13 Mar 2018 22:06:23 +0100 From: "Maciej S. Szmigiero" Subject: [PATCH v3 2/9] x86/microcode/AMD: check whether the equivalence table fits in the file To: Borislav Petkov Cc: Thomas Gleixner , Ingo Molnar , "H. Peter Anvin" , x86@kernel.org, linux-kernel@vger.kernel.org References: Message-ID: Date: Tue, 13 Mar 2018 22:06:23 +0100 User-Agent: Mozilla/5.0 (X11; Linux x86_64; rv:52.0) Gecko/20100101 Thunderbird/52.6.0 MIME-Version: 1.0 In-Reply-To: Content-Type: text/plain; charset=iso-8859-2 Content-Language: en-US Content-Transfer-Encoding: 7bit Sender: linux-kernel-owner@vger.kernel.org Precedence: bulk List-ID: X-Mailing-List: linux-kernel@vger.kernel.org Before loading a CPU equivalence table from a microcode container file we need to verify whether this file is actually large enough to contain the table of a size indicated in this file. If it is not, there is no point of continuing with loading it since microcode patches are located after the equivalence table anyway. Signed-off-by: Maciej S. Szmigiero --- arch/x86/kernel/cpu/microcode/amd.c | 54 ++++++++++++++++++++++++++----------- 1 file changed, 38 insertions(+), 16 deletions(-) diff --git a/arch/x86/kernel/cpu/microcode/amd.c b/arch/x86/kernel/cpu/microcode/amd.c index ffe0d0ce57fc..ac06e2819f26 100644 --- a/arch/x86/kernel/cpu/microcode/amd.c +++ b/arch/x86/kernel/cpu/microcode/amd.c @@ -80,20 +80,29 @@ static u16 find_equiv_id(struct equiv_cpu_entry *equiv_table, u32 sig) * Returns the amount of bytes consumed while scanning. @desc contains all the * data we're going to use in later stages of the application. */ -static ssize_t parse_container(u8 *ucode, ssize_t size, struct cont_desc *desc) +static size_t parse_container(u8 *ucode, size_t size, struct cont_desc *desc) { struct equiv_cpu_entry *eq; - ssize_t orig_size = size; + size_t orig_size = size; u32 *hdr = (u32 *)ucode; + size_t eq_size; u16 eq_id; u8 *buf; /* Am I looking at an equivalence table header? */ + if (size < CONTAINER_HDR_SZ) + return 0; + if (hdr[0] != UCODE_MAGIC || hdr[1] != UCODE_EQUIV_CPU_TABLE_TYPE || hdr[2] == 0) return CONTAINER_HDR_SZ; + eq_size = hdr[2]; + if (eq_size < sizeof(*eq) || + size - CONTAINER_HDR_SZ < eq_size) + return 0; + buf = ucode; eq = (struct equiv_cpu_entry *)(buf + CONTAINER_HDR_SZ); @@ -101,8 +110,8 @@ static ssize_t parse_container(u8 *ucode, ssize_t size, struct cont_desc *desc) /* Find the equivalence ID of our CPU in this table: */ eq_id = find_equiv_id(eq, desc->cpuid_1_eax); - buf += hdr[2] + CONTAINER_HDR_SZ; - size -= hdr[2] + CONTAINER_HDR_SZ; + buf += eq_size + CONTAINER_HDR_SZ; + size -= eq_size + CONTAINER_HDR_SZ; /* * Scan through the rest of the container to find where it ends. We do @@ -159,15 +168,13 @@ static ssize_t parse_container(u8 *ucode, ssize_t size, struct cont_desc *desc) */ static void scan_containers(u8 *ucode, size_t size, struct cont_desc *desc) { - ssize_t rem = size; - - while (rem >= 0) { - ssize_t s = parse_container(ucode, rem, desc); + while (size > 0) { + size_t s = parse_container(ucode, size, desc); if (!s) return; ucode += s; - rem -= s; + size -= s; } } @@ -540,15 +547,30 @@ static enum ucode_state apply_microcode_amd(int cpu) return UCODE_UPDATED; } -static int install_equiv_cpu_table(const u8 *buf) +static int install_equiv_cpu_table(const u8 *buf, size_t buf_size) { unsigned int *ibuf = (unsigned int *)buf; - unsigned int type = ibuf[1]; - unsigned int size = ibuf[2]; + unsigned int type, size; + + if (buf_size < CONTAINER_HDR_SZ) { + pr_err("no container header\n"); + return -EINVAL; + } + + type = ibuf[1]; + if (type != UCODE_EQUIV_CPU_TABLE_TYPE) { + pr_err("invalid type field in container file section header\n"); + return -EINVAL; + } + + size = ibuf[2]; + if (size < sizeof(struct equiv_cpu_entry)) { + pr_err("equivalent CPU table too short\n"); + return -EINVAL; + } - if (type != UCODE_EQUIV_CPU_TABLE_TYPE || !size) { - pr_err("empty section/" - "invalid type field in container file section header\n"); + if (buf_size - CONTAINER_HDR_SZ < size) { + pr_err("equivalent CPU table truncated\n"); return -EINVAL; } @@ -655,7 +677,7 @@ static enum ucode_state __load_microcode_amd(u8 family, const u8 *data, int crnt_size = 0; int offset; - offset = install_equiv_cpu_table(data); + offset = install_equiv_cpu_table(data, size); if (offset < 0) { pr_err("failed to create equivalent cpu table\n"); return ret;