xref: /freebsd-src/contrib/opencsd/decoder/include/mem_acc/trc_mem_acc_base.h (revision c120c5646da1a1d2c4d90fd069a7e2a8d559eb46)
1*c120c564SAndrew Turner /*!
2*c120c564SAndrew Turner  * \file       trc_mem_acc_base.h
3*c120c564SAndrew Turner  * \brief      OpenCSD : Memory accessor base class.
4*c120c564SAndrew Turner  *
5*c120c564SAndrew Turner  * \copyright  Copyright (c) 2015, ARM Limited. All Rights Reserved.
6*c120c564SAndrew Turner  */
7*c120c564SAndrew Turner 
8*c120c564SAndrew Turner /*
9*c120c564SAndrew Turner  * Redistribution and use in source and binary forms, with or without modification,
10*c120c564SAndrew Turner  * are permitted provided that the following conditions are met:
11*c120c564SAndrew Turner  *
12*c120c564SAndrew Turner  * 1. Redistributions of source code must retain the above copyright notice,
13*c120c564SAndrew Turner  * this list of conditions and the following disclaimer.
14*c120c564SAndrew Turner  *
15*c120c564SAndrew Turner  * 2. Redistributions in binary form must reproduce the above copyright notice,
16*c120c564SAndrew Turner  * this list of conditions and the following disclaimer in the documentation
17*c120c564SAndrew Turner  * and/or other materials provided with the distribution.
18*c120c564SAndrew Turner  *
19*c120c564SAndrew Turner  * 3. Neither the name of the copyright holder nor the names of its contributors
20*c120c564SAndrew Turner  * may be used to endorse or promote products derived from this software without
21*c120c564SAndrew Turner  * specific prior written permission.
22*c120c564SAndrew Turner  *
23*c120c564SAndrew Turner  * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS 'AS IS' AND
24*c120c564SAndrew Turner  * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED
25*c120c564SAndrew Turner  * WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED.
26*c120c564SAndrew Turner  * IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT,
27*c120c564SAndrew Turner  * INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES
28*c120c564SAndrew Turner  * (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES;
29*c120c564SAndrew Turner  * LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND
30*c120c564SAndrew Turner  * ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
31*c120c564SAndrew Turner  * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS
32*c120c564SAndrew Turner  * SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
33*c120c564SAndrew Turner  */
34*c120c564SAndrew Turner 
35*c120c564SAndrew Turner #ifndef ARM_TRC_MEM_ACC_BASE_H_INCLUDED
36*c120c564SAndrew Turner #define ARM_TRC_MEM_ACC_BASE_H_INCLUDED
37*c120c564SAndrew Turner 
38*c120c564SAndrew Turner #include "opencsd/ocsd_if_types.h"
39*c120c564SAndrew Turner #include <string>
40*c120c564SAndrew Turner 
41*c120c564SAndrew Turner /*!
42*c120c564SAndrew Turner  * @class TrcMemAccessorBase
43*c120c564SAndrew Turner  * @brief Memory range to access by trace decoder.
44*c120c564SAndrew Turner  *
45*c120c564SAndrew Turner  * Represents a memory access range for the trace decoder.
46*c120c564SAndrew Turner  * Range inclusive from m_startAddress to m_endAddress.
47*c120c564SAndrew Turner  * e.g. a 1k range from 0x1000 has start of 0x1000 and end of 0x13FF
48*c120c564SAndrew Turner  *
49*c120c564SAndrew Turner  * Derived classes provide specific access types such as binary files and memory buffers.
50*c120c564SAndrew Turner  *
51*c120c564SAndrew Turner  */
52*c120c564SAndrew Turner class TrcMemAccessorBase
53*c120c564SAndrew Turner {
54*c120c564SAndrew Turner public:
55*c120c564SAndrew Turner 
56*c120c564SAndrew Turner     /** Describes the storage type of the underlying memory accessor */
57*c120c564SAndrew Turner     enum MemAccTypes {
58*c120c564SAndrew Turner         MEMACC_UNKNOWN,
59*c120c564SAndrew Turner         MEMACC_FILE,        //<! Binary data file accessor
60*c120c564SAndrew Turner         MEMACC_BUFPTR,      //<! memory buffer accessor
61*c120c564SAndrew Turner         MEMACC_CB_IF,       //<! callback interface accessor - use for live memory access
62*c120c564SAndrew Turner     };
63*c120c564SAndrew Turner 
64*c120c564SAndrew Turner     /** default constructor */
65*c120c564SAndrew Turner     TrcMemAccessorBase(MemAccTypes type);
66*c120c564SAndrew Turner 
67*c120c564SAndrew Turner     /** costruct with address range values */
68*c120c564SAndrew Turner     TrcMemAccessorBase(MemAccTypes type, ocsd_vaddr_t startAddr, ocsd_vaddr_t endAddr);
69*c120c564SAndrew Turner 
70*c120c564SAndrew Turner     /** default desctructor */
~TrcMemAccessorBase()71*c120c564SAndrew Turner     virtual ~TrcMemAccessorBase() {};
72*c120c564SAndrew Turner 
73*c120c564SAndrew Turner     /*!
74*c120c564SAndrew Turner      * Set the inclusive address range of this accessor.
75*c120c564SAndrew Turner      *
76*c120c564SAndrew Turner      * @param startAddr : start address of the range.
77*c120c564SAndrew Turner      * @param endAddr : end address of the range.
78*c120c564SAndrew Turner      */
79*c120c564SAndrew Turner     void setRange(ocsd_vaddr_t startAddr, ocsd_vaddr_t endAddr);
80*c120c564SAndrew Turner 
81*c120c564SAndrew Turner     /*!
82*c120c564SAndrew Turner      * test if an address is in the inclusive range for this accessor
83*c120c564SAndrew Turner      *
84*c120c564SAndrew Turner      * @param s_address : Address to test.
85*c120c564SAndrew Turner      *
86*c120c564SAndrew Turner      * @return const bool  : true if the address is in range.
87*c120c564SAndrew Turner      */
88*c120c564SAndrew Turner     virtual const bool addrInRange(const ocsd_vaddr_t s_address) const;
89*c120c564SAndrew Turner 
90*c120c564SAndrew Turner 
91*c120c564SAndrew Turner     /*!
92*c120c564SAndrew Turner      * test if an address is the start of range for this accessor
93*c120c564SAndrew Turner      *
94*c120c564SAndrew Turner      * @param s_address : Address to test.
95*c120c564SAndrew Turner      *
96*c120c564SAndrew Turner      * @return const bool  : true if the address is start of range.
97*c120c564SAndrew Turner      */
98*c120c564SAndrew Turner     virtual const bool addrStartOfRange(const ocsd_vaddr_t s_address) const;
99*c120c564SAndrew Turner 
100*c120c564SAndrew Turner     /*!
101*c120c564SAndrew Turner      * Test number of bytes available from the start address, up to the number of requested bytes.
102*c120c564SAndrew Turner      * Tests if all the requested bytes are available from the supplied start address.
103*c120c564SAndrew Turner      * Returns the number available up to full requested amount.
104*c120c564SAndrew Turner      *
105*c120c564SAndrew Turner      * @param s_address : Start address within the range.
106*c120c564SAndrew Turner      * @param reqBytes : Number of bytes needed from the start address.
107*c120c564SAndrew Turner      *
108*c120c564SAndrew Turner      * @return const uint32_t  : Bytes available, up to reqBytes. 0 is s_address not in range.
109*c120c564SAndrew Turner      */
110*c120c564SAndrew Turner     virtual const uint32_t bytesInRange(const ocsd_vaddr_t s_address, const uint32_t reqBytes) const;
111*c120c564SAndrew Turner 
112*c120c564SAndrew Turner     /*!
113*c120c564SAndrew Turner      * test is supplied range accessor overlaps this range.
114*c120c564SAndrew Turner      *
115*c120c564SAndrew Turner      * @param *p_test_acc : Accessor to test for overlap.
116*c120c564SAndrew Turner      *
117*c120c564SAndrew Turner      * @return bool  : true if overlap, false if not.
118*c120c564SAndrew Turner      */
119*c120c564SAndrew Turner     virtual const bool overLapRange(const TrcMemAccessorBase *p_test_acc) const;
120*c120c564SAndrew Turner 
121*c120c564SAndrew Turner     /*!
122*c120c564SAndrew Turner      * Read bytes from via the accessor from the memory range.
123*c120c564SAndrew Turner      *
124*c120c564SAndrew Turner      * @param s_address : Start address of the read.
125*c120c564SAndrew Turner      * @param memSpace  : memory space for this access.
126*c120c564SAndrew Turner      * @param trcID     : Trace ID of trace source.
127*c120c564SAndrew Turner      * @param reqBytes : Number of bytes required.
128*c120c564SAndrew Turner      * @param *byteBuffer : Buffer to copy the bytes into.
129*c120c564SAndrew Turner      *
130*c120c564SAndrew Turner      * @return uint32_t : Number of bytes read, 0 if s_address out of range, or mem space not accessible.
131*c120c564SAndrew Turner      */
132*c120c564SAndrew Turner     virtual const uint32_t readBytes(const ocsd_vaddr_t s_address, const ocsd_mem_space_acc_t memSpace, const uint8_t trcID, const uint32_t reqBytes, uint8_t *byteBuffer) = 0;
133*c120c564SAndrew Turner 
134*c120c564SAndrew Turner     /*!
135*c120c564SAndrew Turner      * Validate the address range - ensure addresses aligned, different, st < en etc.
136*c120c564SAndrew Turner      *
137*c120c564SAndrew Turner      * @return bool : true if valid range.
138*c120c564SAndrew Turner      */
139*c120c564SAndrew Turner     virtual const bool validateRange();
140*c120c564SAndrew Turner 
141*c120c564SAndrew Turner 
getType()142*c120c564SAndrew Turner     const enum MemAccTypes getType() const { return m_type; };
143*c120c564SAndrew Turner 
144*c120c564SAndrew Turner     /* handle memory spaces */
setMemSpace(ocsd_mem_space_acc_t memSpace)145*c120c564SAndrew Turner     void setMemSpace(ocsd_mem_space_acc_t memSpace) { m_mem_space = memSpace; };
getMemSpace()146*c120c564SAndrew Turner     const ocsd_mem_space_acc_t getMemSpace() const { return m_mem_space; };
inMemSpace(const ocsd_mem_space_acc_t mem_space)147*c120c564SAndrew Turner     const bool inMemSpace(const ocsd_mem_space_acc_t mem_space) const { return (bool)(((uint8_t)m_mem_space & (uint8_t)mem_space) != 0); };
148*c120c564SAndrew Turner 
149*c120c564SAndrew Turner     /* memory access info logging */
150*c120c564SAndrew Turner     virtual void getMemAccString(std::string &accStr) const;
151*c120c564SAndrew Turner 
152*c120c564SAndrew Turner protected:
153*c120c564SAndrew Turner     ocsd_vaddr_t m_startAddress;   /**< accessible range start address */
154*c120c564SAndrew Turner     ocsd_vaddr_t m_endAddress;     /**< accessible range end address */
155*c120c564SAndrew Turner     const MemAccTypes m_type;       /**< memory accessor type */
156*c120c564SAndrew Turner     ocsd_mem_space_acc_t m_mem_space;
157*c120c564SAndrew Turner };
158*c120c564SAndrew Turner 
TrcMemAccessorBase(MemAccTypes accType,ocsd_vaddr_t startAddr,ocsd_vaddr_t endAddr)159*c120c564SAndrew Turner inline TrcMemAccessorBase::TrcMemAccessorBase(MemAccTypes accType, ocsd_vaddr_t startAddr, ocsd_vaddr_t endAddr) :
160*c120c564SAndrew Turner      m_startAddress(startAddr),
161*c120c564SAndrew Turner      m_endAddress(endAddr),
162*c120c564SAndrew Turner      m_type(accType),
163*c120c564SAndrew Turner      m_mem_space(OCSD_MEM_SPACE_ANY)
164*c120c564SAndrew Turner {
165*c120c564SAndrew Turner }
166*c120c564SAndrew Turner 
TrcMemAccessorBase(MemAccTypes accType)167*c120c564SAndrew Turner inline TrcMemAccessorBase::TrcMemAccessorBase(MemAccTypes accType) :
168*c120c564SAndrew Turner      m_startAddress(0),
169*c120c564SAndrew Turner      m_endAddress(0),
170*c120c564SAndrew Turner      m_type(accType),
171*c120c564SAndrew Turner      m_mem_space(OCSD_MEM_SPACE_ANY)
172*c120c564SAndrew Turner {
173*c120c564SAndrew Turner }
174*c120c564SAndrew Turner 
setRange(ocsd_vaddr_t startAddr,ocsd_vaddr_t endAddr)175*c120c564SAndrew Turner inline void TrcMemAccessorBase::setRange(ocsd_vaddr_t startAddr, ocsd_vaddr_t endAddr)
176*c120c564SAndrew Turner {
177*c120c564SAndrew Turner      m_startAddress = startAddr;
178*c120c564SAndrew Turner      m_endAddress = endAddr;
179*c120c564SAndrew Turner }
180*c120c564SAndrew Turner 
addrInRange(const ocsd_vaddr_t s_address)181*c120c564SAndrew Turner inline const bool TrcMemAccessorBase::addrInRange(const ocsd_vaddr_t s_address) const
182*c120c564SAndrew Turner {
183*c120c564SAndrew Turner     return (s_address >= m_startAddress) && (s_address <= m_endAddress);
184*c120c564SAndrew Turner }
185*c120c564SAndrew Turner 
addrStartOfRange(const ocsd_vaddr_t s_address)186*c120c564SAndrew Turner inline const bool TrcMemAccessorBase::addrStartOfRange(const ocsd_vaddr_t s_address) const
187*c120c564SAndrew Turner {
188*c120c564SAndrew Turner     return (s_address == m_startAddress);
189*c120c564SAndrew Turner }
190*c120c564SAndrew Turner 
191*c120c564SAndrew Turner 
bytesInRange(const ocsd_vaddr_t s_address,const uint32_t reqBytes)192*c120c564SAndrew Turner inline const uint32_t TrcMemAccessorBase::bytesInRange(const ocsd_vaddr_t s_address, const uint32_t reqBytes) const
193*c120c564SAndrew Turner {
194*c120c564SAndrew Turner     ocsd_vaddr_t bytesInRange = 0;
195*c120c564SAndrew Turner     if(addrInRange(s_address))  // start not in range, return 0.
196*c120c564SAndrew Turner     {
197*c120c564SAndrew Turner         // bytes available till end address.
198*c120c564SAndrew Turner         bytesInRange = m_endAddress - s_address + 1;
199*c120c564SAndrew Turner         if(bytesInRange > reqBytes)
200*c120c564SAndrew Turner             bytesInRange = reqBytes;
201*c120c564SAndrew Turner     }
202*c120c564SAndrew Turner     return (uint32_t)bytesInRange;
203*c120c564SAndrew Turner }
204*c120c564SAndrew Turner 
overLapRange(const TrcMemAccessorBase * p_test_acc)205*c120c564SAndrew Turner inline const bool TrcMemAccessorBase::overLapRange(const TrcMemAccessorBase *p_test_acc) const
206*c120c564SAndrew Turner {
207*c120c564SAndrew Turner     if( addrInRange(p_test_acc->m_startAddress) ||
208*c120c564SAndrew Turner         addrInRange(p_test_acc->m_endAddress)
209*c120c564SAndrew Turner         )
210*c120c564SAndrew Turner         return true;
211*c120c564SAndrew Turner     return false;
212*c120c564SAndrew Turner }
213*c120c564SAndrew Turner 
validateRange()214*c120c564SAndrew Turner inline const bool TrcMemAccessorBase::validateRange()
215*c120c564SAndrew Turner {
216*c120c564SAndrew Turner     if(m_startAddress & 0x1) // at least hword aligned for thumb
217*c120c564SAndrew Turner         return false;
218*c120c564SAndrew Turner     if((m_endAddress + 1) & 0x1)
219*c120c564SAndrew Turner         return false;
220*c120c564SAndrew Turner     if(m_startAddress == m_endAddress) // zero length range.
221*c120c564SAndrew Turner         return false;
222*c120c564SAndrew Turner     if(m_startAddress > m_endAddress) // values bakcwards  /  invalid
223*c120c564SAndrew Turner         return false;
224*c120c564SAndrew Turner     return true;
225*c120c564SAndrew Turner }
226*c120c564SAndrew Turner 
227*c120c564SAndrew Turner 
228*c120c564SAndrew Turner class TrcMemAccFactory
229*c120c564SAndrew Turner {
230*c120c564SAndrew Turner public:
231*c120c564SAndrew Turner     /** Accessor Creation */
232*c120c564SAndrew Turner     static ocsd_err_t CreateBufferAccessor(TrcMemAccessorBase **pAccessor, const ocsd_vaddr_t s_address, const uint8_t *p_buffer, const uint32_t size);
233*c120c564SAndrew Turner     static ocsd_err_t CreateFileAccessor(TrcMemAccessorBase **pAccessor, const std::string &pathToFile, ocsd_vaddr_t startAddr, size_t offset = 0, size_t size = 0);
234*c120c564SAndrew Turner     static ocsd_err_t CreateCBAccessor(TrcMemAccessorBase **pAccessor, const ocsd_vaddr_t s_address, const ocsd_vaddr_t e_address, const ocsd_mem_space_acc_t mem_space);
235*c120c564SAndrew Turner 
236*c120c564SAndrew Turner     /** Accessor Destruction */
237*c120c564SAndrew Turner     static void DestroyAccessor(TrcMemAccessorBase *pAccessor);
238*c120c564SAndrew Turner private:
TrcMemAccFactory()239*c120c564SAndrew Turner     TrcMemAccFactory() {};
~TrcMemAccFactory()240*c120c564SAndrew Turner     ~TrcMemAccFactory() {};
241*c120c564SAndrew Turner };
242*c120c564SAndrew Turner 
243*c120c564SAndrew Turner #endif // ARM_TRC_MEM_ACC_BASE_H_INCLUDED
244*c120c564SAndrew Turner 
245*c120c564SAndrew Turner /* End of File trc_mem_acc_base.h */
246