From: Andrii Anisov <andrii.anisov@globallogic.com>
To: Oleksandr Tyshchenko <oleksandr.tyshchenko@globallogic.com>
Cc: Ian Campbell <ian.campbell@citrix.com>,
Stefano Stabellini <stefano.stabellini@eu.citrix.com>,
Ian Jackson <Ian.Jackson@eu.citrix.com>,
Julien Grall <julien.grall@linaro.org>, Tim Deegan <tim@xen.org>,
"xen-devel@lists.xen.org" <xen-devel@lists.xen.org>
Subject: Re: [PATCH v4] xen/tools: Introduce QNX IFS loader
Date: Thu, 25 Sep 2014 18:55:52 +0300 [thread overview]
Message-ID: <CAGQvs6jgPXMuiQTCWYid-ki4FWX-h=m0LENy9Hz55NfzHzdfuA@mail.gmail.com> (raw)
In-Reply-To: <1411573423-32327-2-git-send-email-oleksandr.tyshchenko@globallogic.com>
Acked-by: Andrii Anisov <andrii.anisov@globallogic.com>
Andrii Anisov | Team Lead
GlobalLogic
Kyiv, 03038, Protasov Business Park, M.Grinchenka, 2/1
P +38.044.492.9695x3664 M +380505738852 S andriyanisov
www.globallogic.com
http://www.globallogic.com/email_disclaimer.txt
On Wed, Sep 24, 2014 at 6:43 PM, Oleksandr Tyshchenko
<oleksandr.tyshchenko@globallogic.com> wrote:
> This patch was developed according to instruction:
> http://www.qnx.com/developers/docs/6.4.1/neutrino/building/load_process.html
>
> Add ability to load QNX IFS image. The loader probe function
> based on "Writing an IPL Program" howto from qnx.com
> and performs image validation and startup entry point location.
> Because of the fact that the image is already in place some
> customizations have been done. Also in case of multiple images
> (very seldom case) only the first image will be loaded.
>
> There are some restrictions:
> 1. The base address of the image (image attribute) and the location
> of RAM in system must be synchronized with RAM base address in Xen.
> 2. The QNX IFS image must be created as a simple binary image.
>
> Signed-off-by: Oleksandr Tyshchenko <oleksandr.tyshchenko@globallogic.com>
> Cc: Ian Campbell <ian.campbell@citrix.com>
> Cc: Julien Grall <julien.grall@linaro.org>
> ---
> tools/libxc/Makefile | 1 +
> tools/libxc/xc_dom_qnxifsloader.c | 228 ++++++++++++++++++++++++++++++++++++++
> 2 files changed, 229 insertions(+)
> create mode 100644 tools/libxc/xc_dom_qnxifsloader.c
>
> diff --git a/tools/libxc/Makefile b/tools/libxc/Makefile
> index 2cca2b2..1462f53 100644
> --- a/tools/libxc/Makefile
> +++ b/tools/libxc/Makefile
> @@ -66,6 +66,7 @@ GUEST_SRCS-y += xc_dom_elfloader.c
> GUEST_SRCS-$(CONFIG_X86) += xc_dom_bzimageloader.c
> GUEST_SRCS-$(CONFIG_X86) += xc_dom_decompress_lz4.c
> GUEST_SRCS-$(CONFIG_ARM) += xc_dom_armzimageloader.c
> +GUEST_SRCS-$(CONFIG_ARM) += xc_dom_qnxifsloader.c
> GUEST_SRCS-y += xc_dom_binloader.c
> GUEST_SRCS-y += xc_dom_compat_linux.c
>
> diff --git a/tools/libxc/xc_dom_qnxifsloader.c b/tools/libxc/xc_dom_qnxifsloader.c
> new file mode 100644
> index 0000000..8b03885
> --- /dev/null
> +++ b/tools/libxc/xc_dom_qnxifsloader.c
> @@ -0,0 +1,228 @@
> +/*
> + * Xen domain builder -- QNX IFS bits
> + *
> + * Parse and load QNX IFS image.
> + *
> + * Copyright (C) 2014, Globallogic.
> + *
> + * This library is free software; you can redistribute it and/or
> + * modify it under the terms of the GNU Lesser General Public
> + * License as published by the Free Software Foundation;
> + * version 2.1 of the License.
> + *
> + * This library 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
> + * Lesser General Public License for more details.
> + *
> + * You should have received a copy of the GNU Lesser General Public
> + * License along with this library; if not, write to the Free Software
> + * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA
> + *
> + */
> +
> +#include <stdio.h>
> +#include <stdlib.h>
> +#include <inttypes.h>
> +
> +#include "xg_private.h"
> +#include "xc_dom.h"
> +
> +struct startup_header {
> + uint32_t signature; /* Header sig */
> + uint16_t version; /* Header vers */
> + uint8_t flags1; /* Misc flags */
> + uint8_t flags2; /* No flags defined yet */
> + uint16_t header_size; /* sizeof(struct startup_header) */
> + uint16_t machine; /* Machine type */
> + uint32_t startup_vaddr; /* Virtual Address to transfer */
> + /* to after IPL is done */
> + uint32_t paddr_bias; /* Value to add to physical address */
> + /* to get a value to put into a */
> + /* pointer and indirected through */
> + uint32_t image_paddr; /* Physical address of image */
> + uint32_t ram_paddr; /* Physical address of RAM to copy */
> + /* image to (startup_size bytes copied) */
> + uint32_t ram_size; /* Amount of RAM used by the startup */
> + /* program and executables contained */
> + /* in the file system */
> + uint32_t startup_size; /* Size of startup (never compressed) */
> + uint32_t stored_size; /* Size of entire image */
> + uint32_t imagefs_paddr; /* Set by IPL to where the imagefs is */
> + /* when startup runs */
> + uint32_t imagefs_size; /* Size of uncompressed imagefs */
> + uint16_t preboot_size; /* Size of loaded before header */
> + uint16_t zero0; /* Zeros */
> + uint32_t zero[3]; /* Zeros */
> + uint32_t info[48]; /* Array of startup_info* structures */
> +};
> +
> +#define STARTUP_HDR_SIGNATURE 0x00ff7eeb
> +
> +static int calc_checksum(uint32_t *ptr, uint32_t size)
> +{
> + int sum = 0;
> +
> + while ( size > 0 )
> + {
> + sum += *ptr++;
> + size -= 4;
> + }
> +
> + return sum;
> +}
> +
> +static int xc_dom_probe_qnx_ifs(struct xc_dom_image *dom)
> +{
> + struct startup_header *startup_hdr;
> +
> + if ( dom->kernel_blob == NULL )
> + {
> + xc_dom_panic(dom->xch, XC_INTERNAL_ERROR,
> + "%s: no QNX IFS loaded", __FUNCTION__);
> + return -EINVAL;
> + }
> +
> + /* Performs a preliminary checks for a valid image size */
> + if ( dom->kernel_size < sizeof(struct startup_header) )
> + {
> + xc_dom_printf(dom->xch, "%s: QNX IFS is too small", __FUNCTION__);
> + return -EINVAL;
> + }
> +
> + /*
> + * Performs a check for a valid OS signature. We assume that nothing
> + * preceded by startup header because we expect a simple binary image.
> + */
> + startup_hdr = (struct startup_header *)dom->kernel_blob;
> + if ( (startup_hdr->signature != STARTUP_HDR_SIGNATURE) ||
> + (startup_hdr->preboot_size != 0) )
> + {
> + xc_dom_printf(dom->xch, "%s: image is not a QNX IFS", __FUNCTION__);
> + return -EINVAL;
> + }
> +
> + /* Performs a check for a valid startup header size */
> + if ( startup_hdr->header_size != sizeof(struct startup_header) )
> + {
> + xc_dom_printf(dom->xch, "%s: QNX IFS has wrong startup header size", __FUNCTION__);
> + return -EINVAL;
> + }
> +
> + /* Performs a check for a valid stored size */
> + if ( startup_hdr->stored_size != dom->kernel_size )
> + {
> + xc_dom_printf(dom->xch, "%s: QNX IFS has wrong stored size", __FUNCTION__);
> + return -EINVAL;
> + }
> +
> + /* Performs a check for a valid startup size */
> + if ( startup_hdr->startup_size >= startup_hdr->stored_size )
> + {
> + xc_dom_printf(dom->xch, "%s: QNX IFS has wrong startup size", __FUNCTION__);
> + return -EINVAL;
> + }
> +
> + /* Performs a checksums on the startup and the OS image filesystem */
> + if ( (calc_checksum((uint32_t *)startup_hdr, startup_hdr->startup_size) != 0) ||
> + (calc_checksum((uint32_t *)startup_hdr + startup_hdr->startup_size/4,
> + startup_hdr->stored_size - startup_hdr->startup_size) != 0) )
> + {
> + xc_dom_printf(dom->xch, "%s: QNX IFS has wrong checksum", __FUNCTION__);
> + return -EINVAL;
> + }
> +
> + return 0;
> +}
> +
> +static int xc_dom_parse_qnx_ifs(struct xc_dom_image *dom)
> +{
> + struct startup_header *startup_hdr;
> + uint64_t v_start, v_end;
> + uint64_t rambase = dom->rambase_pfn << XC_PAGE_SHIFT;
> +
> + DOMPRINTF_CALLED(dom->xch);
> +
> + /* Do not load kernel at the very first RAM address */
> + v_start = rambase + 0x8000;
> +
> + if ( dom->kernel_size > UINT64_MAX - v_start )
> + {
> + xc_dom_printf(dom->xch, "%s: QNX IFS is too big", __FUNCTION__);
> + return -EINVAL;
> + }
> +
> + v_end = v_start + dom->kernel_size;
> +
> + /* find kernel segment */
> + dom->kernel_seg.vstart = v_start;
> + dom->kernel_seg.vend = v_end;
> +
> + startup_hdr = (struct startup_header *)dom->kernel_blob;
> +
> + /* Performs a sanity check for a valid startup entry point */
> + if ( (startup_hdr->startup_vaddr < v_start) ||
> + (startup_hdr->startup_vaddr > v_end) )
> + {
> + xc_dom_printf(dom->xch, "%s: QNX IFS has wrong startup entry point", __FUNCTION__);
> + return -EINVAL;
> + }
> +
> + dom->parms.virt_entry = startup_hdr->startup_vaddr;
> + dom->parms.virt_base = rambase;
> +
> + dom->guest_type = "xen-3.0-armv7l";
> + DOMPRINTF("%s: %s: RAM starts at %"PRI_xen_pfn,
> + __FUNCTION__, dom->guest_type, dom->rambase_pfn);
> + DOMPRINTF("%s: %s: 0x%" PRIx64 " -> 0x%" PRIx64 "",
> + __FUNCTION__, dom->guest_type,
> + dom->kernel_seg.vstart, dom->kernel_seg.vend);
> +
> + return 0;
> +}
> +
> +static int xc_dom_load_qnx_ifs(struct xc_dom_image *dom)
> +{
> + void *dst;
> +
> + DOMPRINTF_CALLED(dom->xch);
> +
> + dst = xc_dom_seg_to_ptr(dom, &dom->kernel_seg);
> + if ( dst == NULL )
> + {
> + DOMPRINTF("%s: xc_dom_seg_to_ptr(dom, &dom->kernel_seg) => NULL",
> + __func__);
> + return -1;
> + }
> +
> + DOMPRINTF("%s: kernel seg %#"PRIx64"-%#"PRIx64,
> + __func__, dom->kernel_seg.vstart, dom->kernel_seg.vend);
> + DOMPRINTF("%s: copy %zd bytes from blob %p to dst %p",
> + __func__, dom->kernel_size, dom->kernel_blob, dst);
> +
> + memcpy(dst, dom->kernel_blob, dom->kernel_size);
> +
> + return 0;
> +}
> +
> +static struct xc_dom_loader qnx_ifs_loader = {
> + .name = "QNX IFS",
> + .probe = xc_dom_probe_qnx_ifs,
> + .parser = xc_dom_parse_qnx_ifs,
> + .loader = xc_dom_load_qnx_ifs,
> +};
> +
> +static void __init register_loader(void)
> +{
> + xc_dom_register_loader(&qnx_ifs_loader);
> +}
> +
> +/*
> + * Local variables:
> + * mode: C
> + * c-file-style: "BSD"
> + * c-basic-offset: 4
> + * tab-width: 4
> + * indent-tabs-mode: nil
> + * End:
> + */
> --
> 1.9.1
>
>
> _______________________________________________
> Xen-devel mailing list
> Xen-devel@lists.xen.org
> http://lists.xen.org/xen-devel
next prev parent reply other threads:[~2014-09-25 15:55 UTC|newest]
Thread overview: 10+ messages / expand[flat|nested] mbox.gz Atom feed top
2014-09-24 15:43 [PATCH v4] xen/tools: Introduce QNX IFS loader Oleksandr Tyshchenko
2014-09-24 15:43 ` Oleksandr Tyshchenko
2014-09-25 10:25 ` Ian Campbell
2014-09-25 15:51 ` Oleksandr Tyshchenko
2014-09-25 15:54 ` Andrii Tseglytskyi
2014-09-25 15:55 ` Andrii Anisov [this message]
2014-09-26 14:37 ` Ian Jackson
2014-09-26 16:21 ` Oleksandr Tyshchenko
2014-09-26 16:35 ` Ian Jackson
2014-09-26 16:47 ` Oleksandr Tyshchenko
Reply instructions:
You may reply publicly to this message via plain-text email
using any one of the following methods:
* Save the following mbox file, import it into your mail client,
and reply-to-all from there: mbox
Avoid top-posting and favor interleaved quoting:
https://en.wikipedia.org/wiki/Posting_style#Interleaved_style
* Reply using the --to, --cc, and --in-reply-to
switches of git-send-email(1):
git send-email \
--in-reply-to='CAGQvs6jgPXMuiQTCWYid-ki4FWX-h=m0LENy9Hz55NfzHzdfuA@mail.gmail.com' \
--to=andrii.anisov@globallogic.com \
--cc=Ian.Jackson@eu.citrix.com \
--cc=ian.campbell@citrix.com \
--cc=julien.grall@linaro.org \
--cc=oleksandr.tyshchenko@globallogic.com \
--cc=stefano.stabellini@eu.citrix.com \
--cc=tim@xen.org \
--cc=xen-devel@lists.xen.org \
/path/to/YOUR_REPLY
https://kernel.org/pub/software/scm/git/docs/git-send-email.html
* If your mail client supports setting the In-Reply-To header
via mailto: links, try the mailto: link
Be sure your reply has a Subject: header at the top and a blank line
before the message body.
This is a public inbox, see mirroring instructions
for how to clone and mirror all data and code used for this inbox;
as well as URLs for NNTP newsgroup(s).