mirror of
https://github.com/opnsense/src.git
synced 2026-06-26 17:10:12 -04:00
To support cc -pg on arm64 we need to implement .mcount. As clang and
gcc think it is function like it just needs to load the arguments
to _mcount and call it.
On gcc the first argument is passed in x0, however this is missing on
clang so we need to load it from the stack. As it's the caller return
address this will be at a known location.
PR: 262709
Reviewed by: emaste (earlier version)
Sponsored by: The FreeBSD Foundation
Differential Revision: https://reviews.freebsd.org/D34634
(cherry picked from commit 16d5f9a164)
123 lines
3.4 KiB
C
123 lines
3.4 KiB
C
/*-
|
|
* SPDX-License-Identifier: MIT-CMU
|
|
*
|
|
* Copyright (c) 1994, 1995, 1996 Carnegie-Mellon University.
|
|
* All rights reserved.
|
|
*
|
|
* Author: Chris G. Demetriou
|
|
*
|
|
* Permission to use, copy, modify and distribute this software and
|
|
* its documentation is hereby granted, provided that both the copyright
|
|
* notice and this permission notice appear in all copies of the
|
|
* software, derivative works or modified versions, and any portions
|
|
* thereof, and that both notices appear in supporting documentation.
|
|
*
|
|
* CARNEGIE MELLON ALLOWS FREE USE OF THIS SOFTWARE IN ITS "AS IS"
|
|
* CONDITION. CARNEGIE MELLON DISCLAIMS ANY LIABILITY OF ANY KIND
|
|
* FOR ANY DAMAGES WHATSOEVER RESULTING FROM THE USE OF THIS SOFTWARE.
|
|
*
|
|
* Carnegie Mellon requests users of this software to return to
|
|
*
|
|
* Software Distribution Coordinator or Software.Distribution@CS.CMU.EDU
|
|
* School of Computer Science
|
|
* Carnegie Mellon University
|
|
* Pittsburgh PA 15213-3890
|
|
*
|
|
* any improvements or extensions that they make and grant Carnegie the
|
|
* rights to redistribute these changes.
|
|
*
|
|
* from: NetBSD: profile.h,v 1.9 1997/04/06 08:47:37 cgd Exp
|
|
* from: FreeBSD: src/sys/alpha/include/profile.h,v 1.4 1999/12/29
|
|
* $FreeBSD$
|
|
*/
|
|
|
|
#ifndef _MACHINE_PROFILE_H_
|
|
#define _MACHINE_PROFILE_H_
|
|
|
|
#if !defined(_KERNEL) && !defined(_SYS_CDEFS_H_)
|
|
#error this file needs sys/cdefs.h as a prerequisite
|
|
#endif
|
|
|
|
#define FUNCTION_ALIGNMENT 32
|
|
|
|
typedef u_long fptrdiff_t;
|
|
|
|
#ifdef _KERNEL
|
|
|
|
#include <machine/cpufunc.h>
|
|
|
|
#define _MCOUNT_DECL void mcount
|
|
#define MCOUNT
|
|
|
|
#define MCOUNT_DECL(s) register_t s;
|
|
#define MCOUNT_ENTER(s) {s = intr_disable(); }
|
|
#define MCOUNT_EXIT(s) {intr_restore(s); }
|
|
|
|
void bintr(void);
|
|
void btrap(void);
|
|
void eintr(void);
|
|
void user(void);
|
|
|
|
#define MCOUNT_FROMPC_USER(pc) \
|
|
((pc < (uintfptr_t)VM_MAXUSER_ADDRESS) ? (uintfptr_t)user : pc)
|
|
|
|
#define MCOUNT_FROMPC_INTR(pc) \
|
|
((pc >= (uintfptr_t)btrap && pc < (uintfptr_t)eintr) ? \
|
|
((pc >= (uintfptr_t)bintr) ? (uintfptr_t)bintr : \
|
|
(uintfptr_t)btrap) : ~0UL)
|
|
|
|
void mcount(uintfptr_t frompc, uintfptr_t selfpc);
|
|
|
|
#else /* !_KERNEL */
|
|
|
|
typedef __uintfptr_t uintfptr_t;
|
|
|
|
#define _MCOUNT_DECL \
|
|
static void _mcount(uintfptr_t frompc, uintfptr_t selfpc) __used; \
|
|
static void _mcount
|
|
|
|
#ifdef __GNUCLIKE_ASM
|
|
/*
|
|
* Call into _mcount. On arm64 the .mcount is a function so callers will
|
|
* handle caller saved registers. As we don't directly touch any callee
|
|
* saved registers we can just load the two arguments and use a tail call
|
|
* into the MI _mcount function.
|
|
*
|
|
* When building with gcc frompc will be in x0, however this is not the
|
|
* case on clang. As such we need to load it from the stack. As long as
|
|
* the caller follows the ABI this will load the correct value.
|
|
*/
|
|
#define MCOUNT __asm( \
|
|
" .text \n" \
|
|
" .align 6 \n" \
|
|
" .type .mcount,#function \n" \
|
|
" .globl .mcount \n" \
|
|
" .mcount: \n" \
|
|
" .cfi_startproc \n" \
|
|
/* Load the caller return address as frompc */ \
|
|
" ldr x0, [x29, #8] \n" \
|
|
/* Use our return address as selfpc */ \
|
|
" mov x1, lr \n" \
|
|
" b _mcount \n" \
|
|
" .cfi_endproc \n" \
|
|
" .size .mcount, . - .mcount \n" \
|
|
);
|
|
#if 0
|
|
/*
|
|
* If clang passed frompc correctly we could implement it like this, however
|
|
* all clang versions we care about would need to be fixed before we could
|
|
* make this change.
|
|
*/
|
|
void
|
|
mcount(uintfptr_t frompc)
|
|
{
|
|
_mcount(frompc, __builtin_return_address(0));
|
|
}
|
|
#endif
|
|
#else
|
|
#define MCOUNT
|
|
#endif
|
|
|
|
#endif /* _KERNEL */
|
|
|
|
#endif /* !_MACHINE_PROFILE_H_ */
|