/* * Copyright (c) 2014-2015 Dmitry V. Levin * All rights reserved. * * Redistribution and use in source and binary forms, with or without * modification, are permitted provided that the following conditions * are met: * 1. Redistributions of source code must retain the above copyright * notice, this list of conditions and the following disclaimer. * 2. Redistributions in binary form must reproduce the above copyright * notice, this list of conditions and the following disclaimer in the * documentation and/or other materials provided with the distribution. * 3. The name of the author may not be used to endorse or promote products * derived from this software without specific prior written permission. * * THIS SOFTWARE IS PROVIDED BY THE AUTHOR ``AS IS'' AND ANY EXPRESS OR * IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES * OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED. * IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR ANY DIRECT, INDIRECT, * INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT * NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, * DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY * THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF * THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. */ #include "defs.h" #include "xlat/kexec_load_flags.h" #include "xlat/kexec_arch_values.h" #ifndef KEXEC_ARCH_MASK # define KEXEC_ARCH_MASK 0xffff0000 #endif #ifndef KEXEC_SEGMENT_MAX # define KEXEC_SEGMENT_MAX 16 #endif static void print_seg(const unsigned long *seg) { tprints("{"); printaddr(seg[0]); tprintf(", %lu, ", seg[1]); printaddr(seg[2]); tprintf(", %lu}", seg[3]); } static void print_kexec_segments(struct tcb *tcp, const unsigned long addr, const unsigned long len) { unsigned long seg[4]; const size_t sizeof_seg = ARRAY_SIZE(seg) * current_wordsize; unsigned int i; if (!len) { tprints("[]"); return; } if (len > KEXEC_SEGMENT_MAX) { printaddr(addr); return; } if (umove_ulong_array_or_printaddr(tcp, addr, seg, ARRAY_SIZE(seg))) return; tprints("["); print_seg(seg); for (i = 1; i < len; ++i) { tprints(", "); if (umove_ulong_array_or_printaddr(tcp, addr + i * sizeof_seg, seg, ARRAY_SIZE(seg))) break; print_seg(seg); } tprints("]"); } SYS_FUNC(kexec_load) { unsigned long n; /* entry, nr_segments */ printaddr(tcp->u_arg[0]); tprintf(", %lu, ", tcp->u_arg[1]); /* segments */ print_kexec_segments(tcp, tcp->u_arg[2], tcp->u_arg[1]); tprints(", "); /* flags */ n = tcp->u_arg[3]; printxval(kexec_arch_values, n & KEXEC_ARCH_MASK, "KEXEC_ARCH_???"); n &= ~KEXEC_ARCH_MASK; if (n) { tprints("|"); printflags(kexec_load_flags, n, "KEXEC_???"); } return RVAL_DECODED; } #include "xlat/kexec_file_load_flags.h" SYS_FUNC(kexec_file_load) { /* kernel_fd */ printfd(tcp, tcp->u_arg[0]); tprints(", "); /* initrd_fd */ printfd(tcp, tcp->u_arg[1]); tprints(", "); /* cmdline_len */ tprintf("%lu, ", tcp->u_arg[2]); /* cmdline */ printstr(tcp, tcp->u_arg[3], tcp->u_arg[2]); tprints(", "); /* flags */ printflags(kexec_file_load_flags, tcp->u_arg[4], "KEXEC_FILE_???"); return RVAL_DECODED; }