| /* Target-dependent code for Motorola 68000 BSD's. | 
 |  | 
 |    Copyright (C) 2004-2024 Free Software Foundation, Inc. | 
 |  | 
 |    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 <http://www.gnu.org/licenses/>.  */ | 
 |  | 
 | #include "arch-utils.h" | 
 | #include "frame.h" | 
 | #include "osabi.h" | 
 | #include "regcache.h" | 
 | #include "regset.h" | 
 | #include "trad-frame.h" | 
 | #include "tramp-frame.h" | 
 | #include "gdbtypes.h" | 
 |  | 
 | #include "m68k-tdep.h" | 
 | #include "solib-svr4.h" | 
 |  | 
 | /* Core file support.  */ | 
 |  | 
 | /* Sizeof `struct reg' in <machine/reg.h>.  */ | 
 | #define M68KBSD_SIZEOF_GREGS	(18 * 4) | 
 |  | 
 | /* Sizeof `struct fpreg' in <machine/reg.h.  */ | 
 | #define M68KBSD_SIZEOF_FPREGS	(((8 * 3) + 3) * 4) | 
 |  | 
 | int | 
 | m68kbsd_fpreg_offset (struct gdbarch *gdbarch, int regnum) | 
 | { | 
 |   int fp_len = gdbarch_register_type (gdbarch, regnum)->length (); | 
 |    | 
 |   if (regnum >= M68K_FPC_REGNUM) | 
 |     return 8 * fp_len + (regnum - M68K_FPC_REGNUM) * 4; | 
 |  | 
 |   return (regnum - M68K_FP0_REGNUM) * fp_len; | 
 | } | 
 |  | 
 | /* Supply register REGNUM from the buffer specified by FPREGS and LEN | 
 |    in the floating-point register set REGSET to register cache | 
 |    REGCACHE.  If REGNUM is -1, do this for all registers in REGSET.  */ | 
 |  | 
 | static void | 
 | m68kbsd_supply_fpregset (const struct regset *regset, | 
 | 			 struct regcache *regcache, | 
 | 			 int regnum, const void *fpregs, size_t len) | 
 | { | 
 |   struct gdbarch *gdbarch = regcache->arch (); | 
 |   const gdb_byte *regs = (const gdb_byte *) fpregs; | 
 |   int i; | 
 |  | 
 |   gdb_assert (len >= M68KBSD_SIZEOF_FPREGS); | 
 |  | 
 |   for (i = M68K_FP0_REGNUM; i <= M68K_PC_REGNUM; i++) | 
 |     { | 
 |       if (regnum == i || regnum == -1) | 
 | 	regcache->raw_supply (i, regs + m68kbsd_fpreg_offset (gdbarch, i)); | 
 |     } | 
 | } | 
 |  | 
 | /* Supply register REGNUM from the buffer specified by GREGS and LEN | 
 |    in the general-purpose register set REGSET to register cache | 
 |    REGCACHE.  If REGNUM is -1, do this for all registers in REGSET.  */ | 
 |  | 
 | static void | 
 | m68kbsd_supply_gregset (const struct regset *regset, | 
 | 			struct regcache *regcache, | 
 | 			int regnum, const void *gregs, size_t len) | 
 | { | 
 |   const gdb_byte *regs = (const gdb_byte *) gregs; | 
 |   int i; | 
 |  | 
 |   gdb_assert (len >= M68KBSD_SIZEOF_GREGS); | 
 |  | 
 |   for (i = M68K_D0_REGNUM; i <= M68K_PC_REGNUM; i++) | 
 |     { | 
 |       if (regnum == i || regnum == -1) | 
 | 	regcache->raw_supply (i, regs + i * 4); | 
 |     } | 
 |  | 
 |   if (len >= M68KBSD_SIZEOF_GREGS + M68KBSD_SIZEOF_FPREGS) | 
 |     { | 
 |       regs += M68KBSD_SIZEOF_GREGS; | 
 |       len -= M68KBSD_SIZEOF_GREGS; | 
 |       m68kbsd_supply_fpregset (regset, regcache, regnum, regs, len); | 
 |     } | 
 | } | 
 |  | 
 | /* Motorola 68000 register sets.  */ | 
 |  | 
 | static const struct regset m68kbsd_gregset = | 
 | { | 
 |   NULL, | 
 |   m68kbsd_supply_gregset, | 
 |   NULL, | 
 |   REGSET_VARIABLE_SIZE | 
 | }; | 
 |  | 
 | static const struct regset m68kbsd_fpregset = | 
 | { | 
 |   NULL, | 
 |   m68kbsd_supply_fpregset | 
 | }; | 
 |  | 
 | /* Iterate over core file register note sections.  */ | 
 |  | 
 | static void | 
 | m68kbsd_iterate_over_regset_sections (struct gdbarch *gdbarch, | 
 | 				      iterate_over_regset_sections_cb *cb, | 
 | 				      void *cb_data, | 
 | 				      const struct regcache *regcache) | 
 | { | 
 |   cb (".reg", M68KBSD_SIZEOF_GREGS, M68KBSD_SIZEOF_GREGS, &m68kbsd_gregset, | 
 |       NULL, cb_data); | 
 |   cb (".reg2", M68KBSD_SIZEOF_FPREGS, M68KBSD_SIZEOF_FPREGS, &m68kbsd_fpregset, | 
 |       NULL, cb_data); | 
 | } | 
 |  | 
 |  | 
 | static void | 
 | m68kbsd_init_abi (struct gdbarch_info info, struct gdbarch *gdbarch) | 
 | { | 
 |   m68k_gdbarch_tdep *tdep = gdbarch_tdep<m68k_gdbarch_tdep> (gdbarch); | 
 |  | 
 |   tdep->jb_pc = 5; | 
 |   tdep->jb_elt_size = 4; | 
 |  | 
 |   set_gdbarch_decr_pc_after_break (gdbarch, 2); | 
 |  | 
 |   set_gdbarch_iterate_over_regset_sections | 
 |     (gdbarch, m68kbsd_iterate_over_regset_sections); | 
 |  | 
 |   /* NetBSD ELF uses the SVR4 ABI.  */ | 
 |   m68k_svr4_init_abi (info, gdbarch); | 
 |   tdep->struct_return = pcc_struct_return; | 
 |  | 
 |   /* NetBSD ELF uses SVR4-style shared libraries.  */ | 
 |   set_solib_svr4_fetch_link_map_offsets | 
 |     (gdbarch, svr4_ilp32_fetch_link_map_offsets); | 
 | } | 
 |  | 
 | void _initialize_m68kbsd_tdep (); | 
 | void | 
 | _initialize_m68kbsd_tdep () | 
 | { | 
 |   gdbarch_register_osabi (bfd_arch_m68k, 0, GDB_OSABI_NETBSD, | 
 | 			  m68kbsd_init_abi); | 
 | } |