Skip to content

Commit

Permalink
arm: support Thumb-1 with CONFIG_SYS_THUMB_BUILD
Browse files Browse the repository at this point in the history
When building a Thumb-1-only target with CONFIG_SYS_THUMB_BUILD,
some files fail to build, most of the time because they include
mcr instructions, which only exist for Thumb-2.

This patch introduces a Kconfig option CONFIG_THUMB2 and uses
it to select between Thumb-2 and ARM mode for the aforementioned
files.

Signed-off-by: Albert ARIBAUD <[email protected]>
  • Loading branch information
albert-aribaud-u-boot committed Nov 10, 2015
1 parent bf104ff commit 62e9207
Show file tree
Hide file tree
Showing 12 changed files with 154 additions and 3 deletions.
2 changes: 2 additions & 0 deletions Makefile
Original file line number Diff line number Diff line change
Expand Up @@ -605,6 +605,8 @@ KBUILD_CFLAGS += $(KCFLAGS)
UBOOTINCLUDE := \
-Iinclude \
$(if $(KBUILD_SRC), -I$(srctree)/include) \
$(if $(CONFIG_SYS_THUMB_BUILD), $(if $(CONFIG_HAS_THUMB2),, \
-I$(srctree)/arch/$(ARCH)/thumb1/include),) \
-I$(srctree)/arch/$(ARCH)/include \
-include $(srctree)/include/linux/kconfig.h

Expand Down
5 changes: 5 additions & 0 deletions arch/arm/Kconfig
Original file line number Diff line number Diff line change
Expand Up @@ -10,6 +10,9 @@ config ARM64
config HAS_VBAR
bool

config HAS_THUMB2
bool

config CPU_ARM720T
bool

Expand All @@ -32,9 +35,11 @@ config CPU_ARM1176
config CPU_V7
bool
select HAS_VBAR
select HAS_THUMB2

config CPU_V7M
bool
select HAS_THUMB2

config CPU_PXA
bool
Expand Down
11 changes: 11 additions & 0 deletions arch/arm/cpu/arm926ejs/Makefile
Original file line number Diff line number Diff line change
Expand Up @@ -20,3 +20,14 @@ obj-$(CONFIG_MX25) += mx25/
obj-$(CONFIG_MX27) += mx27/
obj-$(if $(filter mxs,$(SOC)),y) += mxs/
obj-$(if $(filter spear,$(SOC)),y) += spear/

# some files can only build in ARM or THUMB2, not THUMB1

ifdef CONFIG_SYS_THUMB_BUILD
ifndef CONFIG_HAS_THUMB2

CFLAGS_cpu.o := -marm
CFLAGS_cache.o := -marm

endif
endif
5 changes: 5 additions & 0 deletions arch/arm/cpu/arm926ejs/cache.c
Original file line number Diff line number Diff line change
Expand Up @@ -82,4 +82,9 @@ void flush_dcache_all(void)
/*
* Stub implementations for l2 cache operations
*/

__weak void l2_cache_disable(void) {}

#if defined CONFIG_SYS_THUMB_BUILD
__weak void invalidate_l2_cache(void) {}
#endif
4 changes: 4 additions & 0 deletions arch/arm/include/asm/cache.h
Original file line number Diff line number Diff line change
Expand Up @@ -16,6 +16,9 @@
/*
* Invalidate L2 Cache using co-proc instruction
*/
#ifdef CONFIG_SYS_THUMB_BUILD
void invalidate_l2_cache(void);
#else
static inline void invalidate_l2_cache(void)
{
unsigned int val=0;
Expand All @@ -24,6 +27,7 @@ static inline void invalidate_l2_cache(void)
: : "r" (val) : "cc");
isb();
}
#endif

void l2_cache_enable(void);
void l2_cache_disable(void);
Expand Down
24 changes: 24 additions & 0 deletions arch/arm/lib/Makefile
Original file line number Diff line number Diff line change
Expand Up @@ -60,3 +60,27 @@ obj-$(CONFIG_DEBUG_LL) += debug.o
ifneq (,$(findstring -mabi=aapcs-linux,$(PLATFORM_CPPFLAGS)))
extra-y += eabi_compat.o
endif

# some files can only build in ARM or THUMB2, not THUMB1

ifdef CONFIG_SYS_THUMB_BUILD
ifndef CONFIG_HAS_THUMB2

# for C files, just apend -marm, which will override previous -mthumb*

CFLAGS_cache.o := -marm
CFLAGS_cache-cp15.o := -marm

# For .S, drop -mthumb* and other thumb-related options.
# CFLAGS_REMOVE_* would not have an effet, so AFLAGS_REMOVE_*
# was implemented and is used here.
# Also, define ${target}_NO_THUMB_BUILD for these two targets
# so that the code knows it should not use Thumb.

AFLAGS_REMOVE_memset.o := -mthumb -mthumb-interwork
AFLAGS_REMOVE_memcpy.o := -mthumb -mthumb-interwork
AFLAGS_memset.o := -DMEMSET_NO_THUMB_BUILD
AFLAGS_memcpy.o := -DMEMCPY_NO_THUMB_BUILD

endif
endif
11 changes: 11 additions & 0 deletions arch/arm/lib/cache.c
Original file line number Diff line number Diff line change
Expand Up @@ -88,3 +88,14 @@ phys_addr_t noncached_alloc(size_t size, size_t align)
return next;
}
#endif /* CONFIG_SYS_NONCACHED_MEMORY */

#if defined(CONFIG_SYS_THUMB_BUILD)
void invalidate_l2_cache(void)
{
unsigned int val = 0;

asm volatile("mcr p15, 1, %0, c15, c11, 0 @ invl l2 cache"
: : "r" (val) : "cc");
isb();
}
#endif
4 changes: 2 additions & 2 deletions arch/arm/lib/memcpy.S
Original file line number Diff line number Diff line change
Expand Up @@ -13,7 +13,7 @@
#include <linux/linkage.h>
#include <asm/assembler.h>

#ifdef CONFIG_SYS_THUMB_BUILD
#if defined(CONFIG_SYS_THUMB_BUILD) && !defined(MEMCPY_NO_THUMB_BUILD)
#define W(instr) instr.w
#else
#define W(instr) instr
Expand Down Expand Up @@ -62,7 +62,7 @@

/* Prototype: void *memcpy(void *dest, const void *src, size_t n); */
.syntax unified
#ifdef CONFIG_SYS_THUMB_BUILD
#if defined(CONFIG_SYS_THUMB_BUILD) && !defined(MEMCPY_NO_THUMB_BUILD)
.thumb
.thumb_func
#endif
Expand Down
2 changes: 1 addition & 1 deletion arch/arm/lib/memset.S
Original file line number Diff line number Diff line change
Expand Up @@ -16,7 +16,7 @@
.align 5

.syntax unified
#ifdef CONFIG_SYS_THUMB_BUILD
#if defined(CONFIG_SYS_THUMB_BUILD) && !defined(MEMSET_NO_THUMB_BUILD)
.thumb
.thumb_func
#endif
Expand Down
10 changes: 10 additions & 0 deletions arch/arm/mach-orion5x/Makefile
Original file line number Diff line number Diff line change
Expand Up @@ -16,3 +16,13 @@ obj-y += timer.o
ifndef CONFIG_SKIP_LOWLEVEL_INIT
obj-y += lowlevel_init.o
endif

# some files can only build in ARM or THUMB2, not THUMB1

ifdef CONFIG_SYS_THUMB_BUILD
ifndef CONFIG_HAS_THUMB2

CFLAGS_cpu.o := -marm

endif
endif
69 changes: 69 additions & 0 deletions arch/arm/thumb1/include/asm/proc-armv/system.h
Original file line number Diff line number Diff line change
@@ -0,0 +1,69 @@
/*
* Thumb-1 drop-in for the linux/include/asm-arm/proc-armv/system.h
*
* (C) Copyright 2015
* Albert ARIBAUD <[email protected]>
*
* The original file does not build in Thumb mode. However, in U-Boot
* we don't use interrupt context, so we can redefine these as empty
* memory barriers, which makes Thumb-1 compiler happy.
*
* SPDX-License-Identifier: GPL-2.0+
*/

/*
* Use the same macro name as linux/include/asm-arm/proc-armv/system.h
* here, so that if the original ever gets included after us, it won't
* try to re-redefine anything.
*/

#ifndef __ASM_PROC_SYSTEM_H
#define __ASM_PROC_SYSTEM_H

/*
* Redefine all original macros with static inline functions containing
* a simple memory barrier, so that they produce the same instruction
* ordering constraints as their original counterparts.
* We use static inline functions rather than macros so that we can tell
* the compiler to not complain about unused arguments.
*/

static inline void local_irq_save(
unsigned long flags __attribute__((unused)))
{
__asm__ __volatile__ ("" : : : "memory");
}

static inline void local_irq_enable(void)
{
__asm__ __volatile__ ("" : : : "memory");
}

static inline void local_irq_disable(void)
{
__asm__ __volatile__ ("" : : : "memory");
}

static inline void __stf(void)
{
__asm__ __volatile__ ("" : : : "memory");
}

static inline void __clf(void)
{
__asm__ __volatile__ ("" : : : "memory");
}

static inline void local_save_flags(
unsigned long flags __attribute__((unused)))
{
__asm__ __volatile__ ("" : : : "memory");
}

static inline void local_irq_restore(
unsigned long flags __attribute__((unused)))
{
__asm__ __volatile__ ("" : : : "memory");
}

#endif /* __ASM_PROC_SYSTEM_H */
10 changes: 10 additions & 0 deletions examples/standalone/Makefile
Original file line number Diff line number Diff line change
Expand Up @@ -73,3 +73,13 @@ $(obj)/%.srec: $(obj)/% FORCE
$(obj)/%.bin: OBJCOPYFLAGS := -O binary
$(obj)/%.bin: $(obj)/% FORCE
$(call if_changed,objcopy)

# some files can only build in ARM or THUMB2, not THUMB1

ifdef CONFIG_SYS_THUMB_BUILD
ifndef CONFIG_HAS_THUMB2

CFLAGS_stubs.o := -marm

endif
endif

0 comments on commit 62e9207

Please sign in to comment.