From mboxrd@z Thu Jan 1 00:00:00 1970 Return-Path: Received: from simark.ca by simark.ca with LMTP id Swq7GgSGbGjPVy4AWB0awg (envelope-from ) for ; Mon, 07 Jul 2025 22:44:20 -0400 Authentication-Results: simark.ca; dkim=pass (2048-bit key; unprotected) header.d=linaro.org header.i=@linaro.org header.a=rsa-sha256 header.s=google header.b=AS/LWYJg; dkim-atps=neutral Received: by simark.ca (Postfix, from userid 112) id 545D61E11C; Mon, 7 Jul 2025 22:44:20 -0400 (EDT) X-Spam-Checker-Version: SpamAssassin 4.0.1 (2024-03-25) on simark.ca X-Spam-Level: X-Spam-Status: No, score=-5.8 required=5.0 tests=ARC_SIGNED,ARC_VALID,BAYES_00, DKIM_SIGNED,DKIM_VALID,DKIM_VALID_AU,MAILING_LIST_MULTI, RCVD_IN_DNSWL_MED,RCVD_IN_SBL_CSS,RCVD_IN_VALIDITY_CERTIFIED, RCVD_IN_VALIDITY_RPBL,RCVD_IN_VALIDITY_SAFE autolearn=ham autolearn_force=no version=4.0.1 Received: from server2.sourceware.org (server2.sourceware.org [8.43.85.97]) (using TLSv1.3 with cipher TLS_AES_256_GCM_SHA384 (256/256 bits) key-exchange X25519 server-signature ECDSA (prime256v1) server-digest SHA256) (No client certificate requested) by simark.ca (Postfix) with ESMTPS id EAFF51E0C2 for ; Mon, 7 Jul 2025 22:44:16 -0400 (EDT) Received: from server2.sourceware.org (localhost [IPv6:::1]) by sourceware.org (Postfix) with ESMTP id 7B17E385B53F for ; Tue, 8 Jul 2025 02:44:15 +0000 (GMT) DKIM-Filter: OpenDKIM Filter v2.11.0 sourceware.org 7B17E385B53F Authentication-Results: sourceware.org; dkim=pass (2048-bit key, unprotected) header.d=linaro.org header.i=@linaro.org header.a=rsa-sha256 header.s=google header.b=AS/LWYJg Received: from mail-qv1-xf30.google.com (mail-qv1-xf30.google.com [IPv6:2607:f8b0:4864:20::f30]) by sourceware.org (Postfix) with ESMTPS id 5B6B3385AC1E for ; Tue, 8 Jul 2025 02:43:32 +0000 (GMT) DMARC-Filter: OpenDMARC Filter v1.4.2 sourceware.org 5B6B3385AC1E Authentication-Results: sourceware.org; dmarc=pass (p=none dis=none) header.from=linaro.org Authentication-Results: sourceware.org; spf=pass smtp.mailfrom=linaro.org ARC-Filter: OpenARC Filter v1.0.0 sourceware.org 5B6B3385AC1E Authentication-Results: server2.sourceware.org; arc=none smtp.remote-ip=2607:f8b0:4864:20::f30 ARC-Seal: i=1; a=rsa-sha256; d=sourceware.org; s=key; t=1751942612; cv=none; b=g1Ull+Gm7CIL15eWWqwkgkDHx+p/1qgvyOS8bczQFPtoGw/fQ41AJRPooOdnzWV4j6EU/V5YxdHCpXP9VmLBXx0PR0pCKs7WugHfwJG4fNn3Ql1WVtQCxTu2zwSdEwmq/g9sFQGApgol4OmfGNHGv9yDLj7+qrgVNw7V3dbUIYw= ARC-Message-Signature: i=1; a=rsa-sha256; d=sourceware.org; s=key; t=1751942612; c=relaxed/simple; bh=CLNPRSTVPdDW4Kjm8i0jR3vPFWqhcqwRmZkyPxP0fTU=; h=DKIM-Signature:From:To:Subject:Date:Message-ID:MIME-Version; b=UUy8GCQWOq2xO+R1HEtySrqmtdqPbzkT7y6V/bXRbbV9N2HWrVaDFkgyeRK32U35tjoCL8GPHSvK0jl2eeUY5L0L+NUR6NLT0sQxTpBNJcZuqJA8mWUixb32SznQiqj2f6QDxQ9FM53EI1nIV98zfewpv/su8Drf5T5c4XDYCTw= ARC-Authentication-Results: i=1; server2.sourceware.org DKIM-Filter: OpenDKIM Filter v2.11.0 sourceware.org 5B6B3385AC1E Received: by mail-qv1-xf30.google.com with SMTP id 6a1803df08f44-6fadd3ad18eso41271896d6.2 for ; Mon, 07 Jul 2025 19:43:32 -0700 (PDT) DKIM-Signature: v=1; a=rsa-sha256; c=relaxed/relaxed; d=linaro.org; s=google; t=1751942611; x=1752547411; darn=sourceware.org; h=mime-version:message-id:date:user-agent:references:in-reply-to :subject:cc:to:from:from:to:cc:subject:date:message-id:reply-to; bh=qS9ibsawOFXD/tFdkJ+ixZ+bpu/RnkwVRDL8/E/CKUM=; b=AS/LWYJgeH8zri/TzdvvK7gRkldWsplbz9qhoW67zPs2Me0TzMgI/qVnjmSPde29QO cjtI6jQ9rIuzNoBfbNEgrb8wic+sXykO6PT3gzgw6edgGXkDRMiga4RwiISti73U3qP1 OMO33id7OkjUXLbVKsk0JOuBZ/2fMDfGr7LAavsSIs9LvJtImX42Bi7w/AndzF5oo1Ye grKqWAU5ZEqee2S5A5wuI9TRZf3FrJ6fYEZylKhxl5LBkoxOYciokZzH0DiczTjetTip CucnccZGBbA+4yv6o921/Vjx1GnjdAYCP9mnluVxycE2JM+j103kpZ7xJ470pzsU3yVN qwPA== X-Google-DKIM-Signature: v=1; a=rsa-sha256; c=relaxed/relaxed; d=1e100.net; s=20230601; t=1751942611; x=1752547411; h=mime-version:message-id:date:user-agent:references:in-reply-to :subject:cc:to:from:x-gm-message-state:from:to:cc:subject:date :message-id:reply-to; bh=qS9ibsawOFXD/tFdkJ+ixZ+bpu/RnkwVRDL8/E/CKUM=; b=dwLlPGwW5lTkIn4Uv7T2FUnApN3YduFyPaI/BDqOF76ZVXcexgtuLBcAhjsP6xGKJS u+vfYY9bxwGUwBJr9NmNpH9Rn08obLMyHNKKV1ZqnstgtbTt4G3/ZXdqgCItp3jIgYrf Km8sLBuPeL2BM9vG0nw+oWTpeH+b+UIJDf9v2JIxtTGS56CsS6MihaW3CG0V6+SBb8lq OEWwOC8mqL9DHM1Hc4kE5CS91d7sB3jO4zpXRMNK5zudRyZGPVuYeUkVl4SC+0we6GFu DLA+jyfD4mc01b7Mkth6aKYpanFUkC1CXHr40s97CSYziHL7N+ZGildB/gVAK+AMbr85 7lGg== X-Gm-Message-State: AOJu0YxZwWvQ2ePjiyDrQzuH6nfjWiCpJGCaLNXBEVohCmPbOsRzBf+g aTNRECun/G1dmGwQU1QtLKiMG2djU620jsgvxVqwBf1JJlNAcGjNYeviXmK3c3I9XP8= X-Gm-Gg: ASbGnctwzQJltzsK+dB++d6Zb07N+zZPSu+4KTXc12oHB/6HxnKhQHMJIsuX9xduz4z lJE87iNHzmK4Z2y6VrRNsPnzZSbLYBC7aINGt0AdTBr7kiDp2LEcMQH4KSFEDhsC+r4h7uOJmYT k3hCPSoyj/IJWuUzDN44Nqg+TUTpuh0djYxVBxfGHgmvBL9miGj7UDXIApUU+Zf59YhUU24WWJX AbH1MAhFqMMNu4CmGdMIs+yBd3G6Qxpxy/q8omkbZpX0LURcptEglLWxSK4pbAoWCKbFI0PGSSp Ws6BZeY9mJ4Nz4Zzf4WLrAmQ+Isy/vAov3y/1FOFy92cHSE1F9y4o7mnKSzmbISUK2o5pkU= X-Google-Smtp-Source: AGHT+IFVKnhGwF+rQGcjTcCb1/2MpeP557i5hxWPQYD00RnkTN6tzjpLZNpRm+fmzF/O90EVbXIkiw== X-Received: by 2002:ad4:5ce9:0:b0:6fa:c31a:af20 with SMTP id 6a1803df08f44-7047d92ced2mr24329826d6.5.1751942611244; Mon, 07 Jul 2025 19:43:31 -0700 (PDT) Received: from localhost ([2804:14d:7e39:88d6:eed:782a:14c2:417a]) by smtp.gmail.com with ESMTPSA id 6a1803df08f44-702c4d601a7sm68108166d6.107.2025.07.07.19.43.29 (version=TLS1_3 cipher=TLS_AES_256_GCM_SHA384 bits=256/256); Mon, 07 Jul 2025 19:43:30 -0700 (PDT) From: Thiago Jung Bauermann To: Tankut Baris Aktemur Cc: gdb-patches@sourceware.org, Markus Metzger Subject: Re: [PATCH v2 06/47] gdb, intelgt: add the target-dependent definitions for the Intel GT architecture In-Reply-To: <20241213-upstream-intelgt-mvp-v2-6-5c4caeb7b33d@intel.com> (Tankut Baris Aktemur's message of "Fri, 13 Dec 2024 16:59:23 +0100") References: <20241213-upstream-intelgt-mvp-v2-0-5c4caeb7b33d@intel.com> <20241213-upstream-intelgt-mvp-v2-6-5c4caeb7b33d@intel.com> User-Agent: mu4e 1.12.11; emacs 30.1 Date: Mon, 07 Jul 2025 23:43:27 -0300 Message-ID: <871pqrtnao.fsf@linaro.org> MIME-Version: 1.0 Content-Type: text/plain X-BeenThere: gdb-patches@sourceware.org X-Mailman-Version: 2.1.30 Precedence: list List-Id: Gdb-patches mailing list List-Unsubscribe: , List-Archive: List-Post: List-Help: List-Subscribe: , Errors-To: gdb-patches-bounces~public-inbox=simark.ca@sourceware.org Hello Baris, Reviewing this patch series has been on my todo list for a long time. I'll try to go through it this week. Tankut Baris Aktemur writes: > Introduce gdb/intelgt-tdep.c for the Intel GT target. The target is > defined in a future patch as a gdbserver low target implementation. > > Other than, for example, X86-64, IntelGT does not have a dedicated I suggest using "Unlike" instead of "Other than" here. > breakpoint instruction. Instead, it has a breakpoint bit in each > instruction. Hence, any instruction can be used as a breakpoint > instruction. > diff --git a/gdb/intelgt-tdep.c b/gdb/intelgt-tdep.c > new file mode 100755 > index 0000000000000000000000000000000000000000..57c359bf355c5771db38b8d213f6681a043c2b33 > --- /dev/null > +++ b/gdb/intelgt-tdep.c > @@ -0,0 +1,980 @@ > +/* Target-dependent code for the Intel(R) Graphics Technology architecture. > + > + Copyright (C) 2019-2024 Free Software Foundation, Inc. Does the copyright on this file really start on 2019? Also, reminder to update to 2025. > + > + This file is part of GDB. > + > + This program is free software; you can redistribute it and/or modify > + it under the terms of the GNU General Public License as published by > + the Free Software Foundation; either version 3 of the License, or > + (at your option) any later version. > + > + This program is distributed in the hope that it will be useful, > + but WITHOUT ANY WARRANTY; without even the implied warranty of > + MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the > + GNU General Public License for more details. > + > + You should have received a copy of the GNU General Public License > + along with this program. If not, see . */ > + > +#include "arch-utils.h" > +#include "arch/intelgt.h" > +#include "cli/cli-cmds.h" > +#include "dwarf2/frame.h" > +#include "frame-unwind.h" > +#include "gdbsupport/gdb_obstack.h" > +#include "gdbtypes.h" > +#include "target.h" > +#include "target-descriptions.h" > +#include "value.h" > +#include "gdbthread.h" > +#include "inferior.h" > +#include "user-regs.h" > +#include > + > +/* Global debug flag. */ > +static bool intelgt_debug = false; > + > +/* Print an "intelgt" debug statement. */ > + > +#define intelgt_debug_printf(fmt, ...) \ > + debug_prefixed_printf_cond (intelgt_debug, "intelgt", fmt, ##__VA_ARGS__) > + > +/* Regnum pair describing the assigned regnum range for a single > + regset. START is inclusive, END is exclusive. */ > + > +struct regnum_range > +{ > + int start; > + int end; > +}; > + > +/* The encoding for XE version enumerates follows this pattern, which is > + aligned with the IGA encoding. */ > + > +#define XE_VERSION(MAJ, MIN) (((MAJ) << 24) | (MIN)) > + > +/* Supported GDB GEN platforms. */ > + > +enum xe_version > +{ > + XE_INVALID = 0, > + XE_HP = XE_VERSION (1, 1), > + XE_HPG = XE_VERSION (1, 2), > + XE_HPC = XE_VERSION (1, 4), > + XE2 = XE_VERSION (2, 0), > +}; > + > +/* Helper functions to request and translate the device id/version. */ > + > +[[maybe_unused]] static xe_version get_xe_version (unsigned int device_id); If the definition of get_xe_version is guarded by "#ifdef HAVE_LIBIGA64", then this prototype isn't needed. Also, the definition of get_xe_version can be moved to patch 8 ("gdb, intelgt: add disassemble feature for the Intel GT architecture."), which uses it. > +/* Data specific for this architecture. */ > + > +struct intelgt_gdbarch_tdep : gdbarch_tdep_base > +{ > + /* $r0 GRF register number. */ > + int r0_regnum = -1; > + > + /* $ce register number in the regcache. */ > + int ce_regnum = -1; > + > + /* Register number for the GRF containing function return value. */ > + int retval_regnum = -1; > + > + /* Register number for the control register. */ > + int cr0_regnum = -1; > + > + /* Register number for the state register. */ > + int sr0_regnum = -1; > + > + /* Register number for the instruction base virtual register. */ > + int isabase_regnum = -1; > + > + /* Register number for the general state base SBA register. */ > + int genstbase_regnum = -1; > + > + /* Register number for the DBG0 register. */ > + int dbg0_regnum = -1; > + > + /* Assigned regnum ranges for DWARF regsets. */ > + regnum_range regset_ranges[intelgt::REGSET_COUNT]; > + > + /* Enabled pseudo-registers for the current target description. */ > + std::vector enabled_pseudo_regs; > + > + /* Cached $framedesc pseudo-register type. */ > + type *framedesc_type = nullptr; > + > + /* Initialize ranges to -1 as "not-yet-set" indicator. */ > + intelgt_gdbarch_tdep () > + { > + memset (®set_ranges, -1, sizeof regset_ranges); > + } > + > + /* Return regnum where frame descriptors are stored. */ > + int framedesc_base_regnum () > + { > + /* For EM_INTELGT frame descriptors are stored at MAX_GRF - 1. */ > + gdb_assert (regset_ranges[intelgt::REGSET_GRF].end > 1); > + return regset_ranges[intelgt::REGSET_GRF].end - 1; > + } > +}; > + > +/* The 'register_type' gdbarch method. */ > + > +static type * > +intelgt_register_type (gdbarch *gdbarch, int regno) > +{ > + type *typ = tdesc_register_type (gdbarch, regno); > + return typ; Minor nit: no need to declare typ, you can directly return the result of the call to tdesc_register_type. > +} > + > +/* Read part of REGNUM at OFFSET into BUFFER. The length of data to > + read is SIZE. Consider using this helper function when reading > + subregisters of CR0, SR0, and R0. */ > + > +static void > +intelgt_read_register_part (readable_regcache *regcache, int regnum, > + size_t offset, > + gdb::array_view buffer, > + const char *error_message) > +{ > + if (regnum == -1) > + error (_("%s Unexpected reg num '-1'."), error_message); > + > + gdbarch *arch = regcache->arch (); > + const char *regname = gdbarch_register_name (arch, regnum); > + int regsize = register_size (arch, regnum); > + > + if (offset + buffer.size () > regsize) > + error (_("%s[%zu:%zu] is outside the range of %s[%d:0]."), > + regname, (offset + buffer.size () - 1), offset, > + regname, (regsize - 1)); Shouldn't this error include error_message as well? > + register_status reg_status > + = regcache->cooked_read_part (regnum, offset, buffer); > + > + if (reg_status == REG_UNAVAILABLE) > + throw_error (NOT_AVAILABLE_ERROR, > + _("%s Register %s (%d) is not available."), > + error_message, regname, regnum); > + > + if (reg_status == REG_UNKNOWN) > + error (_("%s Register %s (%d) is unknown."), error_message, > + regname, regnum); > +} > + > +static int > +intelgt_pseudo_register_num (gdbarch *arch, const char *name); If you move the definition of intelgt_pseudo_register_num here, you won't need this prototype declaration. > +/* Convert a DWARF register number to a GDB register number. This > + function requires the register listing in the target description to > + be in the same order in each regset as the intended DWARF numbering > + order. Currently this always holds true when gdbserver generates > + the target description. */ > + > +static int > +intelgt_dwarf_reg_to_regnum (gdbarch *gdbarch, int num) > +{ > + constexpr int ip = 0; > + constexpr int ce = 1; > + > + /* Register sets follow this format: [START, END), where START is > + inclusive and END is exclusive. */ > + constexpr regnum_range dwarf_nums[intelgt::REGSET_COUNT] = { > + [intelgt::REGSET_SBA] = { 5, 12 }, > + [intelgt::REGSET_GRF] = { 16, 272 }, > + [intelgt::REGSET_ADDR] = { 272, 288 }, > + [intelgt::REGSET_FLAG] = { 288, 304 }, > + [intelgt::REGSET_ACC] = { 304, 320 }, > + [intelgt::REGSET_MME] = { 320, 336 }, > + }; > + > + /* Number of SBA registers. */ > + constexpr size_t sba_dwarf_len = dwarf_nums[intelgt::REGSET_SBA].end > + - dwarf_nums[intelgt::REGSET_SBA].start; > + > + /* Map the DWARF register numbers of SBA registers to their names. > + Base number is dwarf_nums[intelgt::REGSET_SBA].start. */ > + constexpr const char* sba_dwarf_reg_order[sba_dwarf_len] { > + "btbase", > + "scrbase", > + "genstbase", > + "sustbase", > + "blsustbase", > + "blsastbase", > + "scrbase2" > + }; > + > + intelgt_gdbarch_tdep *data > + = gdbarch_tdep (gdbarch); Nit: the above fits in one line, no need to break into two. There are several occurrences of this in the file. I'll just comment on this one. > + if (num == ip) > + return intelgt_pseudo_register_num (gdbarch, "ip"); > + if (num == ce) > + return data->ce_regnum; > + > + for (int regset = 0; regset < intelgt::REGSET_COUNT; ++regset) > + if (num >= dwarf_nums[regset].start && num < dwarf_nums[regset].end) > + { > + if (regset == intelgt::REGSET_SBA) > + { > + /* For SBA registers we first find out the name of the register > + out of DWARF register number and then find the register number > + corresponding to the name. */ > + int sba_num = num - dwarf_nums[intelgt::REGSET_SBA].start; > + const char* name = sba_dwarf_reg_order [sba_num]; > + > + return user_reg_map_name_to_regnum (gdbarch, name, -1); > + } > + else > + { > + int candidate = data->regset_ranges[regset].start + num > + - dwarf_nums[regset].start; > + > + if (candidate < data->regset_ranges[regset].end) > + return candidate; > + } > + } > + > + return -1; > +} > + > +/* Return the PC of the first real instruction. */ > + > +static CORE_ADDR > +intelgt_skip_prologue (gdbarch *gdbarch, CORE_ADDR start_pc) > +{ > + intelgt_debug_printf ("start_pc: %s", paddress (gdbarch, start_pc)); > + CORE_ADDR func_addr; > + > + if (find_pc_partial_function (start_pc, nullptr, &func_addr, nullptr)) > + { > + CORE_ADDR post_prologue_pc > + = skip_prologue_using_sal (gdbarch, func_addr); > + > + intelgt_debug_printf ("post prologue pc: %s", > + paddress (gdbarch, post_prologue_pc)); > + > + if (post_prologue_pc != 0) > + return std::max (start_pc, post_prologue_pc); Considering that both start_pc and post_prologue_pc are CORE_ADDR (an unsigned type), checking whether post_prologue_pc is 0 is redundant: std::max will return start_pc in that case. > + } > + > + /* Could not find the end of prologue using SAL. */ > + return start_pc; > +} > + > +/* Implementation of gdbarch's return_value method. */ > + > +static enum return_value_convention > +intelgt_return_value_as_value (gdbarch *gdbarch, value *function, > + type *valtype, regcache *regcache, > + value **read_value, const gdb_byte *writebuf) > +{ > + gdb_assert_not_reached ("intelgt_return_value_as_value is to be " > + "implemented later."); > +} > + > +/* Callback function to unwind the $framedesc register. */ > + > +static value * > +intelgt_dwarf2_prev_framedesc (const frame_info_ptr &this_frame, > + void **this_cache, int regnum) > +{ > + gdbarch *gdbarch = get_frame_arch (this_frame); > + intelgt_gdbarch_tdep *data > + = gdbarch_tdep (gdbarch); > + > + int actual_regnum = data->framedesc_base_regnum (); > + > + /* Unwind the actual GRF register. */ > + return frame_unwind_register_value (this_frame, actual_regnum); > +} > + > +/* Architecture-specific register state initialization. */ > + > +static void > +intelgt_init_reg (gdbarch *gdbarch, int regnum, dwarf2_frame_state_reg *reg, > + const frame_info_ptr &this_frame) > +{ > + int ip_regnum = intelgt_pseudo_register_num (gdbarch, "ip"); > + int framedesc_regnum = intelgt_pseudo_register_num (gdbarch, "framedesc"); > + > + if (regnum == ip_regnum) > + reg->how = DWARF2_FRAME_REG_RA; > + else if (regnum == gdbarch_sp_regnum (gdbarch)) > + reg->how = DWARF2_FRAME_REG_CFA; > + /* We use special functions to unwind the $framedesc register. */ s/special functions/a special function/ > + else if (regnum == framedesc_regnum) > + { > + reg->how = DWARF2_FRAME_REG_FN; > + reg->loc.fn = intelgt_dwarf2_prev_framedesc; > + } > +} > + > +/* A helper function that returns the value of the ISABASE register. */ > + > +static CORE_ADDR > +intelgt_get_isabase (readable_regcache *regcache) > +{ > + gdbarch *gdbarch = regcache->arch (); > + intelgt_gdbarch_tdep *data > + = gdbarch_tdep (gdbarch); > + > + gdb_assert (data->isabase_regnum != -1); > + > + uint64_t isabase = 0; > + if (regcache->cooked_read (data->isabase_regnum, &isabase) != REG_VALID) > + throw_error (NOT_AVAILABLE_ERROR, > + _("Register %d (isabase) is not available"), > + data->isabase_regnum); > + return isabase; > +} > + > +/* The 'unwind_pc' gdbarch method. */ > + > +static CORE_ADDR > +intelgt_unwind_pc (gdbarch *gdbarch, const frame_info_ptr &next_frame) > +{ > + /* Use ip register here, as IGC uses 32bit values (pc is 64bit). */ > + int ip_regnum = intelgt_pseudo_register_num (gdbarch, "ip"); > + CORE_ADDR prev_ip = frame_unwind_register_unsigned (next_frame, > + ip_regnum); There's no need to break the line above. Everything fits in one line. > + intelgt_debug_printf ("prev_ip: %s", paddress (gdbarch, prev_ip)); > + > + /* Program counter is $ip + $isabase. Read directly from the > + regcache instead of unwinding, as the frame unwind info may > + simply be unavailable. The isabase register does not change > + during kernel execution, so this must be safe. */ > + regcache *regcache = get_thread_regcache (inferior_thread ()); > + CORE_ADDR isabase = intelgt_get_isabase (regcache); > + > + return isabase + prev_ip; > +} > + > +/* Frame unwinding. */ > + > +static void > +intelgt_frame_this_id (const frame_info_ptr &this_frame, > + void **this_prologue_cache, frame_id *this_id) > +{ > + /* FIXME: Assembly-level unwinding for intelgt is not available at > + the moment. Stop at the first frame. */ > + *this_id = outer_frame_id; > +} > + > +static const struct frame_unwind intelgt_unwinder = > + { > + "intelgt prologue", > + NORMAL_FRAME, /* type */ > + default_frame_unwind_stop_reason, /* stop_reason */ > + intelgt_frame_this_id, /* this_id */ > + nullptr, /* prev_register */ > + nullptr, /* unwind_data */ > + default_frame_sniffer, /* sniffer */ > + nullptr, /* dealloc_cache */ > + }; This unwinder doesn't do much. Is it necessary? If so, I suggest a comment explaining why. > + > +/* The memory_insert_breakpoint gdbarch method. */ > + > +static int > +intelgt_memory_insert_breakpoint (gdbarch *gdbarch, bp_target_info *bp) > +{ > + intelgt_debug_printf ("req ip: %s", paddress (gdbarch, > + bp->reqstd_address)); > + > + /* Ensure that we have enough space in the breakpoint. */ > + static_assert (intelgt::MAX_INST_LENGTH <= BREAKPOINT_MAX); > + > + gdb_byte inst[intelgt::MAX_INST_LENGTH]; > + int err = target_read_memory (bp->reqstd_address, inst, > + intelgt::MAX_INST_LENGTH); > + if (err != 0) > + { > + /* We could fall back to reading a full and then a compacted > + instruction but I think we should rather allow short reads than > + having the caller try smaller and smaller sizes. */ > + intelgt_debug_printf ("Failed to read memory at %s (%s).", > + paddress (gdbarch, bp->reqstd_address), > + strerror (err)); > + return err; > + } > + > + bp->placed_address = bp->reqstd_address; > + bp->shadow_len = intelgt::inst_length (inst); > + > + /* Make a copy before we set the breakpoint so we can restore the > + original instruction when removing the breakpoint again. > + > + This isn't strictly necessary but it saves one target access. */ > + memcpy (bp->shadow_contents, inst, bp->shadow_len); > + > + const bool already = intelgt::set_breakpoint (inst); > + if (already) > + { > + /* Warn if the breakpoint bit is already set. > + > + There is still a breakpoint, probably hard-coded, and it should > + still trigger and we're still able to step over it. It's just > + not our breakpoint. */ > + warning (_("Using permanent breakpoint at %s."), > + paddress (gdbarch, bp->placed_address)); > + > + /* There's no need to write the unmodified instruction back. */ > + return 0; > + } > + > + err = target_write_raw_memory (bp->placed_address, inst, bp->shadow_len); > + if (err != 0) > + intelgt_debug_printf ("Failed to insert breakpoint at %s (%s).", > + paddress (gdbarch, bp->placed_address), > + strerror (err)); > + > + return err; > +} > + > +/* The memory_remove_breakpoint gdbarch method. */ > + > +static int > +intelgt_memory_remove_breakpoint (gdbarch *gdbarch, struct bp_target_info *bp) > +{ > + intelgt_debug_printf ("req ip: %s, placed ip: %s", > + paddress (gdbarch, bp->reqstd_address), > + paddress (gdbarch, bp->placed_address)); > + > + /* Warn if we're inserting a permanent breakpoint. */ > + if (intelgt::has_breakpoint (bp->shadow_contents)) > + warning (_("Re-inserting permanent breakpoint at %s."), > + paddress (gdbarch, bp->placed_address)); > + > + /* See comment in mem-break.c on write_inferior_memory. */ I wasn't able to find that comment. Has it been removed from GDB? > + int err = target_write_raw_memory (bp->placed_address, bp->shadow_contents, > + bp->shadow_len); > + if (err != 0) > + intelgt_debug_printf ("Failed to remove breakpoint at %s (%s).", > + paddress (gdbarch, bp->placed_address), > + strerror (err)); > + > + return err; > +} > + > +/* The program_breakpoint_here_p gdbarch method. */ > + > +static bool > +intelgt_program_breakpoint_here_p (gdbarch *gdbarch, CORE_ADDR pc) > +{ > + intelgt_debug_printf ("pc: %s", paddress (gdbarch, pc)); > + > + gdb_byte inst[intelgt::MAX_INST_LENGTH]; > + int err = target_read_memory (pc, inst, intelgt::MAX_INST_LENGTH); > + if (err != 0) > + { > + /* We could fall back to reading a full and then a compacted > + instruction but I think we should rather allow short reads than > + having the caller try smaller and smaller sizes. */ > + intelgt_debug_printf ("Failed to read memory at %s (%s).", > + paddress (gdbarch, pc), strerror (err)); > + return err; > + } > + > + const bool is_bkpt = intelgt::has_breakpoint (inst); > + > + intelgt_debug_printf ("%sbreakpoint found.", is_bkpt ? "" : "no "); > + > + return is_bkpt; > +} > + > +/* The 'breakpoint_kind_from_pc' gdbarch method. > + This is a required gdbarch function. */ > + > +static int > +intelgt_breakpoint_kind_from_pc (gdbarch *gdbarch, CORE_ADDR *pcptr) > +{ > + intelgt_debug_printf ("*pcptr: %s", paddress (gdbarch, *pcptr)); > + > + return intelgt::BP_INSTRUCTION; > +} > + > +/* The 'sw_breakpoint_from_kind' gdbarch method. */ > + > +static const gdb_byte * > +intelgt_sw_breakpoint_from_kind (gdbarch *gdbarch, int kind, int *size) > +{ > + intelgt_debug_printf ("kind: %d", kind); > + > + /* We do not support breakpoint instructions. > + > + We use breakpoint bits in instructions, instead. See > + intelgt_memory_insert_breakpoint. */ > + *size = 0; > + return nullptr; > +} > + > +/* Print one instruction from MEMADDR on INFO->STREAM. */ > + > +static int > +intelgt_print_insn (bfd_vma memaddr, struct disassemble_info *info) > +{ > + /* Disassembler is to be added in a later patch. */ > + return -1; > +} > + > +/* Utility function to look up the pseudo-register number by name. Exact > + amount of pseudo-registers may differ and thus fixed constants can't be > + used for this. */ > + > +static int > +intelgt_pseudo_register_num (gdbarch *arch, const char *name) > +{ > + intelgt_gdbarch_tdep *data > + = gdbarch_tdep (arch); > + > + auto iter = std::find (data->enabled_pseudo_regs.begin (), > + data->enabled_pseudo_regs.end (), name); > + gdb_assert (iter != data->enabled_pseudo_regs.end ()); > + return gdbarch_num_regs (arch) + (iter - data->enabled_pseudo_regs.begin ()); > +} > + > +/* The "read_pc" gdbarch method. */ > + > +static CORE_ADDR > +intelgt_read_pc (readable_regcache *regcache) > +{ > + gdbarch *arch = regcache->arch (); > + intelgt_gdbarch_tdep *data > + = gdbarch_tdep (arch); > + > + /* Instruction pointer is stored in CR0.2. */ > + uint32_t ip; > + intelgt_read_register_part (regcache, data->cr0_regnum, > + sizeof (uint32_t) * 2, > + gdb::make_array_view ((gdb_byte *) &ip, > + sizeof (uint32_t)), > + _("Cannot compute PC.")); > + > + /* Program counter is $ip + $isabase. */ > + CORE_ADDR isabase = intelgt_get_isabase (regcache); > + return isabase + ip; > +} > + > +/* The "write_pc" gdbarch method. */ > + > +static void > +intelgt_write_pc (struct regcache *regcache, CORE_ADDR pc) > +{ > + gdbarch *arch = regcache->arch (); > + /* Program counter is $ip + $isabase, can only modify $ip. Need > + to ensure that the new value fits within $ip modification range > + and propagate the write accordingly. */ > + CORE_ADDR isabase = intelgt_get_isabase (regcache); > + if (pc < isabase || pc > isabase + UINT32_MAX) > + error ("Can't update $pc to value %s, out of range", > + paddress (arch, pc)); The error message should be surrounded by _() to mark it as translatable. > + > + intelgt_gdbarch_tdep *data > + = gdbarch_tdep (arch); > + > + /* Instruction pointer is stored in CR0.2. */ > + uint32_t ip = pc - isabase; > + regcache->cooked_write_part (data->cr0_regnum, sizeof (uint32_t) * 2, > + gdb::make_array_view ((gdb_byte *) &ip, > + sizeof (uint32_t))); > +} > + > +/* Return the name of pseudo-register REGNUM. */ > + > +static const char * > +intelgt_pseudo_register_name (gdbarch *arch, int regnum) > +{ > + intelgt_gdbarch_tdep *data > + = gdbarch_tdep (arch); > + > + int base_num = gdbarch_num_regs (arch); > + if (regnum < base_num > + || regnum >= base_num + data->enabled_pseudo_regs.size ()) > + error ("Invalid pseudo-register regnum %d", regnum); The error message should be surrounded by _() to mark it as translatable. > + return data->enabled_pseudo_regs[regnum - base_num].c_str (); > +} > + > +/* Return the GDB type object for the "standard" data type of data in > + pseudo-register REGNUM. */ > + > +static type * > +intelgt_pseudo_register_type (gdbarch *arch, int regnum) > +{ > + const char *name = intelgt_pseudo_register_name (arch, regnum); > + const struct builtin_type *bt = builtin_type (arch); > + intelgt_gdbarch_tdep *data > + = gdbarch_tdep (arch); > + > + if (strcmp (name, "framedesc") == 0) > + { > + if (data->framedesc_type != nullptr) > + return data->framedesc_type; > + type *frame = arch_composite_type (arch, "frame_desc", TYPE_CODE_STRUCT); > + append_composite_type_field (frame, "return_ip", bt->builtin_uint32); Isn't it better to use a function pointer type? This is what x86_64 and aarch64 do: (gdb) ptype $pc type = void (*)() Though I see further below that pointer and address lengths in gdbarch are set to 64 bits, so perhaps things aren't very straightforward in this architecture... > + append_composite_type_field (frame, "return_callmask", > + bt->builtin_uint32); > + append_composite_type_field (frame, "be_sp", bt->builtin_uint32); > + append_composite_type_field (frame, "be_fp", bt->builtin_uint32); > + append_composite_type_field (frame, "fe_fp", bt->builtin_uint64); > + append_composite_type_field (frame, "fe_sp", bt->builtin_uint64); If the 'p' in the fields above mean "pointer", would a void pointer work? This is what x86_64 and aarch64 do: (gdb) ptype $sp type = void * > + data->framedesc_type = frame; > + return frame; > + } > + else if (strcmp (name, "ip") == 0) > + return bt->builtin_uint32; Same note about using a function pointer type as in return_ip above. > + return nullptr; > +} > +/* Architecture initialization. */ > + > +static gdbarch * > +intelgt_gdbarch_init (gdbarch_info info, gdbarch_list *arches) > +{ > + /* If there is already a candidate, use it. */ > + arches = gdbarch_list_lookup_by_info (arches, &info); > + if (arches != nullptr) > + return arches->gdbarch; > + > + const target_desc *tdesc = info.target_desc; > + gdbarch *gdbarch > + = gdbarch_alloc (&info, > + gdbarch_tdep_up (new intelgt_gdbarch_tdep)); There are some calls to error () below. They'll leak gdbarch. The patch adding disassemble support will also introduce some early returns with calls to gdbarch_free (gdbarch). You could instead use gdbarch_up and release () the value when returning it at the end of this function so that it's automatically freed in case of an early return: gdbarch_up gdbarch_u (gdbarch_alloc (&info, gdbarch_tdep_up (new intelgt_gdbarch_tdep))); gdbarch *gdbarch = gdbarch_u.get (); Then at the end: return gdbarch_u.release (); > + intelgt_gdbarch_tdep *data > + = gdbarch_tdep (gdbarch); > + > + /* Initialize register info. */ > + set_gdbarch_num_regs (gdbarch, 0); > + set_gdbarch_register_name (gdbarch, tdesc_register_name); > + > + if (tdesc_has_registers (tdesc)) > + { > + tdesc_arch_data_up tdesc_data = tdesc_data_alloc (); > + > + /* First assign register numbers to all registers. The > + callback function will record any relevant metadata > + about it in the intelgt_gdbarch_data instance to be > + inspected after. */ > + > + tdesc_use_registers (gdbarch, tdesc, std::move (tdesc_data), > + intelgt_unknown_register_cb); > + > + /* Now check the collected metadata to ensure that all > + mandatory pieces are in place. */ > + > + if (data->ce_regnum == -1) > + error ("Debugging requires $ce provided by the target"); > + if (data->retval_regnum == -1) > + error ("Debugging requires return value register to be provided by " > + "the target"); > + if (data->cr0_regnum == -1) > + error ("Debugging requires control register to be provided by " > + "the target"); > + if (data->sr0_regnum == -1) > + error ("Debugging requires state register to be provided by " > + "the target"); The error messages should be surrounded by _() to mark them as translatable. > + /* Unconditionally enabled pseudo-registers: */ > + data->enabled_pseudo_regs.push_back ("ip"); > + data->enabled_pseudo_regs.push_back ("framedesc"); > + > + set_gdbarch_num_pseudo_regs (gdbarch, data->enabled_pseudo_regs.size ()); > + set_gdbarch_pseudo_register_read_value ( > + gdbarch, intelgt_pseudo_register_read_value); > + set_gdbarch_pseudo_register_write (gdbarch, > + intelgt_pseudo_register_write); > + set_tdesc_pseudo_register_type (gdbarch, intelgt_pseudo_register_type); > + set_tdesc_pseudo_register_name (gdbarch, intelgt_pseudo_register_name); > + set_gdbarch_read_pc (gdbarch, intelgt_read_pc); > + set_gdbarch_write_pc (gdbarch, intelgt_write_pc); > + } > + > + /* Populate gdbarch fields. */ > + set_gdbarch_ptr_bit (gdbarch, 64); > + set_gdbarch_addr_bit (gdbarch, 64); I'm a bit surprised by these, considering that PC is 32 bits. If this is correct, then it's worth adding an explanation about the discrepancy. Is this why PC is uint32 instead of a pointer type? > + set_gdbarch_long_bit (gdbarch, 64); > + > + set_gdbarch_register_type (gdbarch, intelgt_register_type); Considering that intelgt_register_type just calls tdesc_register_type, shouldn't this be set to tdesc_register_type directly? > + set_gdbarch_dwarf2_reg_to_regnum (gdbarch, intelgt_dwarf_reg_to_regnum); > + > + set_gdbarch_skip_prologue (gdbarch, intelgt_skip_prologue); > + set_gdbarch_inner_than (gdbarch, core_addr_greaterthan); > + set_gdbarch_unwind_pc (gdbarch, intelgt_unwind_pc); > + dwarf2_append_unwinders (gdbarch); > + frame_unwind_append_unwinder (gdbarch, &intelgt_unwinder); > + > + set_gdbarch_return_value_as_value (gdbarch, intelgt_return_value_as_value); > + > + set_gdbarch_memory_insert_breakpoint (gdbarch, > + intelgt_memory_insert_breakpoint); > + set_gdbarch_memory_remove_breakpoint (gdbarch, > + intelgt_memory_remove_breakpoint); > + set_gdbarch_program_breakpoint_here_p (gdbarch, > + intelgt_program_breakpoint_here_p); > + set_gdbarch_breakpoint_kind_from_pc (gdbarch, > + intelgt_breakpoint_kind_from_pc); > + set_gdbarch_sw_breakpoint_from_kind (gdbarch, > + intelgt_sw_breakpoint_from_kind); > + dwarf2_frame_set_init_reg (gdbarch, intelgt_init_reg); > + > + /* Disassembly. */ > + set_gdbarch_print_insn (gdbarch, intelgt_print_insn); > + > + return gdbarch; > +} > + > +/* Dump the target specific data for this architecture. */ > + > +static void > +intelgt_dump_tdep (gdbarch *gdbarch, ui_file *file) > +{ > + /* Implement target-specific print output if and > + when gdbarch_tdep is defined for this architecture. */ >From looking at gdbarch_dump in gdb/gdbarch-gen.c, you can just pass nullptr to gdbarch_register instead of having to define a dummy function. Though the comment isn't accurate. There is intelgt_gdbarch_tdep, but perhaps it's not necessary to dump any of its values? > +} > + > +static void > +show_intelgt_debug (ui_file *file, int from_tty, > + cmd_list_element *c, const char *value) > +{ > + gdb_printf (file, _("Intel(R) Graphics Technology debugging is " > + "%s.\n"), value); > +} > + > +void _initialize_intelgt_tdep (); > +void > +_initialize_intelgt_tdep () Update to current style using INIT_GDB_FILE macro. > +{ > + gdbarch_register (bfd_arch_intelgt, intelgt_gdbarch_init, > + intelgt_dump_tdep); > + > + /* Debugging flag. */ > + add_setshow_boolean_cmd ("intelgt", class_maintenance, &intelgt_debug, > + _("Set Intel(R) Graphics Technology debugging."), > + _("Show Intel(R) Graphics Technology debugging."), > + _("When on, Intel(R) Graphics Technology " > + "debugging is enabled."), > + nullptr, > + show_intelgt_debug, > + &setdebuglist, &showdebuglist); > +} -- Thiago