atag.hh revision 7585
17585SAli.Saidi@arm.com/*
27585SAli.Saidi@arm.com * Copyright (c) 2010 ARM Limited
37585SAli.Saidi@arm.com * All rights reserved
47585SAli.Saidi@arm.com *
57585SAli.Saidi@arm.com * The license below extends only to copyright in the software and shall
67585SAli.Saidi@arm.com * not be construed as granting a license to any other intellectual
77585SAli.Saidi@arm.com * property including but not limited to intellectual property relating
87585SAli.Saidi@arm.com * to a hardware implementation of the functionality of the software
97585SAli.Saidi@arm.com * licensed hereunder.  You may use the software subject to the license
107585SAli.Saidi@arm.com * terms below provided that you ensure that this notice is replicated
117585SAli.Saidi@arm.com * unmodified and in its entirety in all distributions of the software,
127585SAli.Saidi@arm.com * modified or unmodified, in source code or in binary form.
137585SAli.Saidi@arm.com *
147585SAli.Saidi@arm.com * Redistribution and use in source and binary forms, with or without
157585SAli.Saidi@arm.com * modification, are permitted provided that the following conditions are
167585SAli.Saidi@arm.com * met: redistributions of source code must retain the above copyright
177585SAli.Saidi@arm.com * notice, this list of conditions and the following disclaimer;
187585SAli.Saidi@arm.com * redistributions in binary form must reproduce the above copyright
197585SAli.Saidi@arm.com * notice, this list of conditions and the following disclaimer in the
207585SAli.Saidi@arm.com * documentation and/or other materials provided with the distribution;
217585SAli.Saidi@arm.com * neither the name of the copyright holders nor the names of its
227585SAli.Saidi@arm.com * contributors may be used to endorse or promote products derived from
237585SAli.Saidi@arm.com * this software without specific prior written permission.
247585SAli.Saidi@arm.com *
257585SAli.Saidi@arm.com * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
267585SAli.Saidi@arm.com * "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
277585SAli.Saidi@arm.com * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
287585SAli.Saidi@arm.com * A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
297585SAli.Saidi@arm.com * OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
307585SAli.Saidi@arm.com * SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
317585SAli.Saidi@arm.com * LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
327585SAli.Saidi@arm.com * DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
337585SAli.Saidi@arm.com * THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
347585SAli.Saidi@arm.com * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
357585SAli.Saidi@arm.com * OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
367585SAli.Saidi@arm.com *
377585SAli.Saidi@arm.com * Authors: Ali Saidi
387585SAli.Saidi@arm.com */
397585SAli.Saidi@arm.com
407585SAli.Saidi@arm.com#ifndef __ARCH_ARM_LINUX_ATAG_HH__
417585SAli.Saidi@arm.com#define __ARCH_ARM_LINUX_ATAG_HH__
427585SAli.Saidi@arm.com
437585SAli.Saidi@arm.com#include <cstring>
447585SAli.Saidi@arm.com#include <string>
457585SAli.Saidi@arm.com#include "base/types.hh"
467585SAli.Saidi@arm.com
477585SAli.Saidi@arm.comenum {
487585SAli.Saidi@arm.com    CoreTag   = 0x54410001,
497585SAli.Saidi@arm.com    MemTag    = 0x54410002,
507585SAli.Saidi@arm.com    RevTag    = 0x54410007,
517585SAli.Saidi@arm.com    SerialTag = 0x54410006,
527585SAli.Saidi@arm.com    CmdTag    = 0x54410009,
537585SAli.Saidi@arm.com    NoneTag   = 0x00000000,
547585SAli.Saidi@arm.com};
557585SAli.Saidi@arm.com
567585SAli.Saidi@arm.comclass AtagHeader
577585SAli.Saidi@arm.com{
587585SAli.Saidi@arm.com  protected:
597585SAli.Saidi@arm.com    uint32_t *storage;
607585SAli.Saidi@arm.com    uint32_t _size;
617585SAli.Saidi@arm.com
627585SAli.Saidi@arm.com  public:
637585SAli.Saidi@arm.com    /** Tag (normally starts with 'T''A' and 16 bits of number */
647585SAli.Saidi@arm.com    virtual uint32_t tag() = 0;
657585SAli.Saidi@arm.com    /** If the header should be 0 size */
667585SAli.Saidi@arm.com    virtual bool null() { return false; }
677585SAli.Saidi@arm.com
687585SAli.Saidi@arm.com    uint32_t size() const { return _size; }
697585SAli.Saidi@arm.com
707585SAli.Saidi@arm.com    AtagHeader(uint32_t s)
717585SAli.Saidi@arm.com        : _size(s)
727585SAli.Saidi@arm.com    {
737585SAli.Saidi@arm.com        storage = new uint32_t[size()];
747585SAli.Saidi@arm.com    }
757585SAli.Saidi@arm.com
767585SAli.Saidi@arm.com    virtual ~AtagHeader()
777585SAli.Saidi@arm.com    {
787585SAli.Saidi@arm.com        delete[] storage;
797585SAli.Saidi@arm.com    }
807585SAli.Saidi@arm.com
817585SAli.Saidi@arm.com    uint32_t copyOut(uint8_t *p)
827585SAli.Saidi@arm.com    {
837585SAli.Saidi@arm.com        storage[0] = null() ? 0 : size();
847585SAli.Saidi@arm.com        storage[1] = tag();
857585SAli.Saidi@arm.com        memcpy(p, storage, size() << 2);
867585SAli.Saidi@arm.com        return size() << 2;
877585SAli.Saidi@arm.com    }
887585SAli.Saidi@arm.com};
897585SAli.Saidi@arm.com
907585SAli.Saidi@arm.comclass AtagCore : public AtagHeader
917585SAli.Saidi@arm.com{
927585SAli.Saidi@arm.com  public:
937585SAli.Saidi@arm.com    static const uint32_t Size = 5;
947585SAli.Saidi@arm.com    uint32_t tag() { return CoreTag; }
957585SAli.Saidi@arm.com
967585SAli.Saidi@arm.com    void flags(uint32_t i) { storage[2] = i; }
977585SAli.Saidi@arm.com    void pagesize(uint32_t i) { storage[3] = i; }
987585SAli.Saidi@arm.com    void rootdev(uint32_t i) { storage[4] = i; }
997585SAli.Saidi@arm.com    AtagCore()
1007585SAli.Saidi@arm.com        : AtagHeader(Size)
1017585SAli.Saidi@arm.com    {}
1027585SAli.Saidi@arm.com};
1037585SAli.Saidi@arm.com
1047585SAli.Saidi@arm.comclass AtagMem : public AtagHeader
1057585SAli.Saidi@arm.com{
1067585SAli.Saidi@arm.com  public:
1077585SAli.Saidi@arm.com    static const uint32_t Size = 4;
1087585SAli.Saidi@arm.com    uint32_t tag() { return MemTag; }
1097585SAli.Saidi@arm.com
1107585SAli.Saidi@arm.com    void memSize(uint32_t i) { storage[2] = i; }
1117585SAli.Saidi@arm.com    void memStart(uint32_t i) { storage[3] = i; }
1127585SAli.Saidi@arm.com    AtagMem()
1137585SAli.Saidi@arm.com        : AtagHeader(Size)
1147585SAli.Saidi@arm.com    {}
1157585SAli.Saidi@arm.com};
1167585SAli.Saidi@arm.com
1177585SAli.Saidi@arm.comclass AtagRev : public AtagHeader
1187585SAli.Saidi@arm.com{
1197585SAli.Saidi@arm.com  public:
1207585SAli.Saidi@arm.com    static const uint32_t Size = 3;
1217585SAli.Saidi@arm.com    uint32_t tag() { return RevTag; }
1227585SAli.Saidi@arm.com
1237585SAli.Saidi@arm.com    void rev(uint32_t i) { storage[2] = i; }
1247585SAli.Saidi@arm.com    AtagRev()
1257585SAli.Saidi@arm.com        : AtagHeader(Size)
1267585SAli.Saidi@arm.com    {}
1277585SAli.Saidi@arm.com};
1287585SAli.Saidi@arm.com
1297585SAli.Saidi@arm.com
1307585SAli.Saidi@arm.comclass AtagSerial : public AtagHeader
1317585SAli.Saidi@arm.com{
1327585SAli.Saidi@arm.com  public:
1337585SAli.Saidi@arm.com    static const uint32_t Size = 4;
1347585SAli.Saidi@arm.com    uint32_t tag() { return SerialTag; }
1357585SAli.Saidi@arm.com
1367585SAli.Saidi@arm.com    void sn(uint64_t i) { storage[2] = (uint32_t)i; storage[3] = i >> 32; }
1377585SAli.Saidi@arm.com    AtagSerial()
1387585SAli.Saidi@arm.com        : AtagHeader(Size)
1397585SAli.Saidi@arm.com    {}
1407585SAli.Saidi@arm.com};
1417585SAli.Saidi@arm.com
1427585SAli.Saidi@arm.comclass AtagCmdline : public AtagHeader
1437585SAli.Saidi@arm.com{
1447585SAli.Saidi@arm.com  public:
1457585SAli.Saidi@arm.com    static const uint32_t Size = 3;
1467585SAli.Saidi@arm.com    uint32_t tag() { return CmdTag; }
1477585SAli.Saidi@arm.com
1487585SAli.Saidi@arm.com    void cmdline(const std::string &s)
1497585SAli.Saidi@arm.com    {
1507585SAli.Saidi@arm.com        // Add one for null terminator
1517585SAli.Saidi@arm.com        int len = s.length() + 1;
1527585SAli.Saidi@arm.com
1537585SAli.Saidi@arm.com        // 2 + ceiling(len/4)
1547585SAli.Saidi@arm.com        _size = 2 + ((len + 3) >> 2);
1557585SAli.Saidi@arm.com
1567585SAli.Saidi@arm.com        delete[] storage;
1577585SAli.Saidi@arm.com        storage = new uint32_t[size()];
1587585SAli.Saidi@arm.com
1597585SAli.Saidi@arm.com        strcpy((char*)&storage[2] , s.c_str());
1607585SAli.Saidi@arm.com    }
1617585SAli.Saidi@arm.com    AtagCmdline()
1627585SAli.Saidi@arm.com        : AtagHeader(Size)
1637585SAli.Saidi@arm.com    {}
1647585SAli.Saidi@arm.com};
1657585SAli.Saidi@arm.com
1667585SAli.Saidi@arm.comclass AtagNone : public AtagHeader
1677585SAli.Saidi@arm.com{
1687585SAli.Saidi@arm.com  public:
1697585SAli.Saidi@arm.com    static const uint32_t Size = 2;
1707585SAli.Saidi@arm.com    virtual bool null() { return true; }
1717585SAli.Saidi@arm.com    uint32_t tag() { return NoneTag; }
1727585SAli.Saidi@arm.com    AtagNone()
1737585SAli.Saidi@arm.com        : AtagHeader(Size)
1747585SAli.Saidi@arm.com    {}
1757585SAli.Saidi@arm.com};
1767585SAli.Saidi@arm.com/*
1777585SAli.Saidi@arm.com//
1787585SAli.Saidi@arm.com// example ARM Linux bootloader code
1797585SAli.Saidi@arm.com// this example is distributed under the BSD licence
1807585SAli.Saidi@arm.com// Code taken from http://www.simtec.co.uk/products/SWLINUX/files/booting_article.html
1817585SAli.Saidi@arm.com///
1827585SAli.Saidi@arm.com
1837585SAli.Saidi@arm.com// list of possible tags
1847585SAli.Saidi@arm.com#define ATAG_NONE       0x00000000
1857585SAli.Saidi@arm.com#define ATAG_CORE       0x54410001
1867585SAli.Saidi@arm.com#define ATAG_MEM        0x54410002
1877585SAli.Saidi@arm.com#define ATAG_VIDEOTEXT  0x54410003
1887585SAli.Saidi@arm.com#define ATAG_RAMDISK    0x54410004
1897585SAli.Saidi@arm.com#define ATAG_INITRD2    0x54420005
1907585SAli.Saidi@arm.com#define ATAG_SERIAL     0x54410006
1917585SAli.Saidi@arm.com#define ATAG_REVISION   0x54410007
1927585SAli.Saidi@arm.com#define ATAG_VIDEOLFB   0x54410008
1937585SAli.Saidi@arm.com#define ATAG_CMDLINE    0x54410009
1947585SAli.Saidi@arm.com
1957585SAli.Saidi@arm.com// structures for each atag
1967585SAli.Saidi@arm.comstruct atag_header {
1977585SAli.Saidi@arm.com        u32 size; // length of tag in words including this header
1987585SAli.Saidi@arm.com        u32 tag;  // tag type
1997585SAli.Saidi@arm.com};
2007585SAli.Saidi@arm.com
2017585SAli.Saidi@arm.comstruct atag_core {
2027585SAli.Saidi@arm.com        u32 flags;
2037585SAli.Saidi@arm.com        u32 pagesize;
2047585SAli.Saidi@arm.com        u32 rootdev;
2057585SAli.Saidi@arm.com};
2067585SAli.Saidi@arm.com
2077585SAli.Saidi@arm.comstruct atag_mem {
2087585SAli.Saidi@arm.com        u32     size;
2097585SAli.Saidi@arm.com        u32     start;
2107585SAli.Saidi@arm.com};
2117585SAli.Saidi@arm.com
2127585SAli.Saidi@arm.comstruct atag_videotext {
2137585SAli.Saidi@arm.com        u8              x;
2147585SAli.Saidi@arm.com        u8              y;
2157585SAli.Saidi@arm.com        u16             video_page;
2167585SAli.Saidi@arm.com        u8              video_mode;
2177585SAli.Saidi@arm.com        u8              video_cols;
2187585SAli.Saidi@arm.com        u16             video_ega_bx;
2197585SAli.Saidi@arm.com        u8              video_lines;
2207585SAli.Saidi@arm.com        u8              video_isvga;
2217585SAli.Saidi@arm.com        u16             video_points;
2227585SAli.Saidi@arm.com};
2237585SAli.Saidi@arm.com
2247585SAli.Saidi@arm.comstruct atag_ramdisk {
2257585SAli.Saidi@arm.com        u32 flags;
2267585SAli.Saidi@arm.com        u32 size;
2277585SAli.Saidi@arm.com        u32 start;
2287585SAli.Saidi@arm.com};
2297585SAli.Saidi@arm.com
2307585SAli.Saidi@arm.comstruct atag_initrd2 {
2317585SAli.Saidi@arm.com        u32 start;
2327585SAli.Saidi@arm.com        u32 size;
2337585SAli.Saidi@arm.com};
2347585SAli.Saidi@arm.com
2357585SAli.Saidi@arm.comstruct atag_serialnr {
2367585SAli.Saidi@arm.com        u32 low;
2377585SAli.Saidi@arm.com        u32 high;
2387585SAli.Saidi@arm.com};
2397585SAli.Saidi@arm.com
2407585SAli.Saidi@arm.comstruct atag_revision {
2417585SAli.Saidi@arm.com        u32 rev;
2427585SAli.Saidi@arm.com};
2437585SAli.Saidi@arm.com
2447585SAli.Saidi@arm.comstruct atag_videolfb {
2457585SAli.Saidi@arm.com        u16             lfb_width;
2467585SAli.Saidi@arm.com        u16             lfb_height;
2477585SAli.Saidi@arm.com        u16             lfb_depth;
2487585SAli.Saidi@arm.com        u16             lfb_linelength;
2497585SAli.Saidi@arm.com        u32             lfb_base;
2507585SAli.Saidi@arm.com        u32             lfb_size;
2517585SAli.Saidi@arm.com        u8              red_size;
2527585SAli.Saidi@arm.com        u8              red_pos;
2537585SAli.Saidi@arm.com        u8              green_size;
2547585SAli.Saidi@arm.com        u8              green_pos;
2557585SAli.Saidi@arm.com        u8              blue_size;
2567585SAli.Saidi@arm.com        u8              blue_pos;
2577585SAli.Saidi@arm.com        u8              rsvd_size;
2587585SAli.Saidi@arm.com        u8              rsvd_pos;
2597585SAli.Saidi@arm.com};
2607585SAli.Saidi@arm.com
2617585SAli.Saidi@arm.comstruct atag_cmdline {
2627585SAli.Saidi@arm.com        char    cmdline[1];
2637585SAli.Saidi@arm.com};
2647585SAli.Saidi@arm.com
2657585SAli.Saidi@arm.comstruct atag {
2667585SAli.Saidi@arm.com        struct atag_header hdr;
2677585SAli.Saidi@arm.com        union {
2687585SAli.Saidi@arm.com                struct atag_core         core;
2697585SAli.Saidi@arm.com                struct atag_mem          mem;
2707585SAli.Saidi@arm.com                struct atag_videotext    videotext;
2717585SAli.Saidi@arm.com                struct atag_ramdisk      ramdisk;
2727585SAli.Saidi@arm.com                struct atag_initrd2      initrd2;
2737585SAli.Saidi@arm.com                struct atag_serialnr     serialnr;
2747585SAli.Saidi@arm.com                struct atag_revision     revision;
2757585SAli.Saidi@arm.com                struct atag_videolfb     videolfb;
2767585SAli.Saidi@arm.com                struct atag_cmdline      cmdline;
2777585SAli.Saidi@arm.com        } u;
2787585SAli.Saidi@arm.com};
2797585SAli.Saidi@arm.com*/
2807585SAli.Saidi@arm.com
2817585SAli.Saidi@arm.com
2827585SAli.Saidi@arm.com#endif // __ARCH_ARM_LINUX_ATAG_HH__
283