[PATCH v2 3/3] ppc64: Add a flag to tell the kernel it's booting from kexec

Scott Wood scottwood at freescale.com
Tue Dec 8 07:42:28 AEDT 2015


On Tue, 2015-10-06 at 17:55 -0500, Scott Wood wrote:
> It needs to know this because the SMP release mechanism for Freescale
> book3e is different from when booting with normal hardware.  In theory
> we could simulate the normal spin table mechanism, but not (easily) at
> the addresses U-Boot put in the device tree -- so there'd need to be
> even more communication between the kernel and kexec to set that up.
> 
> Signed-off-by: Scott Wood <scottwood at freescale.com>
> ---
> v2: Use a device tree property rather than setting a flag in the kernel
> image, as requested by Michael Ellerman.

ping?

Michael, can you review this?  Simon is waiting for another PPC person to
review it.

-Scott

> ---
>  kexec/arch/ppc64/Makefile           |  6 +++
>  kexec/arch/ppc64/fdt.c              | 78
> +++++++++++++++++++++++++++++++++++++
>  kexec/arch/ppc64/include/arch/fdt.h |  9 +++++
>  kexec/arch/ppc64/kexec-elf-ppc64.c  |  7 ++++
>  4 files changed, 100 insertions(+)
>  create mode 100644 kexec/arch/ppc64/fdt.c
>  create mode 100644 kexec/arch/ppc64/include/arch/fdt.h
> 
> diff --git a/kexec/arch/ppc64/Makefile b/kexec/arch/ppc64/Makefile
> index 9a6e475..37cd233 100644
> --- a/kexec/arch/ppc64/Makefile
> +++ b/kexec/arch/ppc64/Makefile
> @@ -1,11 +1,15 @@
>  #
>  # kexec ppc64 (linux booting linux)
>  #
> +include $(srcdir)/kexec/libfdt/Makefile.libfdt
> +
>  ppc64_KEXEC_SRCS =  kexec/arch/ppc64/kexec-elf-rel-ppc64.c
>  ppc64_KEXEC_SRCS += kexec/arch/ppc64/kexec-zImage-ppc64.c
>  ppc64_KEXEC_SRCS += kexec/arch/ppc64/kexec-elf-ppc64.c
>  ppc64_KEXEC_SRCS += kexec/arch/ppc64/kexec-ppc64.c
>  ppc64_KEXEC_SRCS += kexec/arch/ppc64/crashdump-ppc64.c
> +ppc64_KEXEC_SRCS += kexec/arch/ppc64/fdt.c
> +ppc64_KEXEC_SRCS += $(LIBFDT_SRCS:%=kexec/libfdt/%)
>  
>  ppc64_ARCH_REUSE_INITRD =
>  
> @@ -13,6 +17,8 @@ ppc64_FS2DT	    = kexec/fs2dt.c
>  ppc64_FS2DT_INCLUDE = -include $(srcdir)/kexec/arch/ppc64/crashdump-ppc64.h
> \
>                        -include $(srcdir)/kexec/arch/ppc64/kexec-ppc64.h
>  
> +ppc64_CPPFLAGS = -I$(srcdir)/kexec/libfdt
> +
>  dist += kexec/arch/ppc64/Makefile $(ppc64_KEXEC_SRCS)			
> \
>  	kexec/arch/ppc64/kexec-ppc64.h kexec/arch/ppc64/crashdump-ppc64.h \
>  	kexec/arch/ppc64/include/arch/options.h
> diff --git a/kexec/arch/ppc64/fdt.c b/kexec/arch/ppc64/fdt.c
> new file mode 100644
> index 0000000..8bc6d2d
> --- /dev/null
> +++ b/kexec/arch/ppc64/fdt.c
> @@ -0,0 +1,78 @@
> +/*
> + * ppc64 fdt fixups
> + *
> + * Copyright 2015 Freescale Semiconductor, Inc.
> + *
> + * 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 (version 2 of the License).
> + *
> + * 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.
> + */
> +
> +#include <arch/fdt.h>
> +#include <libfdt.h>
> +#include <stdio.h>
> +#include <stdlib.h>
> +
> +/*
> + * Let the kernel know it booted from kexec, as some things (e.g.
> + * secondary CPU release) may work differently.
> + */
> +static int fixup_kexec_prop(void *fdt)
> +{
> +	int err, nodeoffset;
> +
> +	nodeoffset = fdt_subnode_offset(fdt, 0, "chosen");
> +	if (nodeoffset < 0)
> +		nodeoffset = fdt_add_subnode(fdt, 0, "chosen");
> +	if (nodeoffset < 0) {
> +		printf("%s: add /chosen %s\n", __func__,
> +		       fdt_strerror(nodeoffset));
> +		return -1;
> +	}
> +
> +	err = fdt_setprop(fdt, nodeoffset, "linux,booted-from-kexec",
> +			  NULL, 0);
> +	if (err < 0) {
> +		printf("%s: couldn't write linux,booted-from-kexec: %s\n",
> +		       __func__, fdt_strerror(err));
> +		return -1;
> +	}
> +
> +	return 0;
> +}
> +
> +
> +/*
> + * For now, assume that the added content fits in the file.
> + * This should be the case when flattening from /proc/device-tree,
> + * and when passing in a dtb, dtc can be told to add padding.
> + */
> +int fixup_dt(char **fdt, off_t *size)
> +{
> +	int ret;
> +
> +	*size += 4096;
> +	*fdt = realloc(*fdt, *size);
> +	if (!*fdt) {
> +		fprintf(stderr, "%s: out of memory\n", __func__);
> +		return -1;
> +	}
> +
> +	ret = fdt_open_into(*fdt, *fdt, *size);
> +	if (ret < 0) {
> +		fprintf(stderr, "%s: fdt_open_into: %s\n", __func__,
> +			fdt_strerror(ret));
> +		return -1;
> +	}
> +
> +	ret = fixup_kexec_prop(*fdt);
> +	if (ret < 0)
> +		return ret;
> +
> +	return 0;
> +}
> diff --git a/kexec/arch/ppc64/include/arch/fdt.h
> b/kexec/arch/ppc64/include/arch/fdt.h
> new file mode 100644
> index 0000000..14f8be2
> --- /dev/null
> +++ b/kexec/arch/ppc64/include/arch/fdt.h
> @@ -0,0 +1,9 @@
> +#ifndef KEXEC_ARCH_PPC64_FDT
> +#define KEXEC_ARCH_PPC64_FDT
> +
> +#include <sys/types.h>
> +
> +int fixup_dt(char **fdt, off_t *size);
> +
> +#endif
> +
> diff --git a/kexec/arch/ppc64/kexec-elf-ppc64.c b/kexec/arch/ppc64/kexec-elf
> -ppc64.c
> index adcee4c..ddd3de8 100644
> --- a/kexec/arch/ppc64/kexec-elf-ppc64.c
> +++ b/kexec/arch/ppc64/kexec-elf-ppc64.c
> @@ -37,6 +37,8 @@
>  #include "kexec-ppc64.h"
>  #include "../../fs2dt.h"
>  #include "crashdump-ppc64.h"
> +#include <libfdt.h>
> +#include <arch/fdt.h>
>  #include <arch/options.h>
>  
>  uint64_t initrd_base, initrd_size;
> @@ -242,6 +244,11 @@ int elf_ppc64_load(int argc, char **argv, const char
> *buf, off_t len,
>  		/* create from fs2dt */
>  		create_flatten_tree(&seg_buf, &seg_size, cmdline);
>  	}
> +
> +	result = fixup_dt(&seg_buf, &seg_size);
> +	if (result < 0)
> +		return result;
> +
>  	my_dt_offset = add_buffer(info, seg_buf, seg_size, seg_size,
>  				0, 0, max_addr, -1);
>  


More information about the Linuxppc-dev mailing list