blob: 8f11a252619b8d34e8302f8f6f3ff532394d937c [file] [log] [blame]
mleach6c73c832015-11-25 16:23:21 +00001/*!
mike leacha6baf052015-08-20 00:24:15 +01002 * \file trc_mem_acc_base.h
mleach6c73c832015-11-25 16:23:21 +00003 * \brief Reference CoreSight Trace Decoder : Memory accessor base class.
mike leacha6baf052015-08-20 00:24:15 +01004 *
5 * \copyright Copyright (c) 2015, ARM Limited. All Rights Reserved.
6 */
7
8/*
9 * Redistribution and use in source and binary forms, with or without modification,
10 * are permitted provided that the following conditions are met:
11 *
12 * 1. Redistributions of source code must retain the above copyright notice,
13 * this list of conditions and the following disclaimer.
14 *
15 * 2. Redistributions in binary form must reproduce the above copyright notice,
16 * this list of conditions and the following disclaimer in the documentation
17 * and/or other materials provided with the distribution.
18 *
19 * 3. Neither the name of the copyright holder nor the names of its contributors
20 * may be used to endorse or promote products derived from this software without
21 * specific prior written permission.
22 *
23 * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS 'AS IS' AND
24 * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED
25 * WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED.
26 * IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT,
27 * INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES
28 * (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES;
29 * LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND
30 * ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
31 * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS
32 * SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
33 */
34
mike leacha6baf052015-08-20 00:24:15 +010035#ifndef ARM_TRC_MEM_ACC_BASE_H_INCLUDED
36#define ARM_TRC_MEM_ACC_BASE_H_INCLUDED
37
38#include "rctdl_if_types.h"
mleachebc28b02015-12-11 11:15:34 +000039#include <string>
mike leacha6baf052015-08-20 00:24:15 +010040
mleach34d6ee12015-08-21 12:47:32 +010041/*!
42 * @class TrcMemAccessorBase
43 * @brief Memory range to access by trace decoder.
44 *
45 * Represents a memory access range for the trace decoder.
46 * Range inclusive from m_startAddress to m_endAddress.
47 * e.g. a 1k range from 0x1000 has start of 0x1000 and end of 0x13FF
48 *
49 * Derived classes provide specific access types such as binary files and memory buffers.
50 *
51 */
mike leacha6baf052015-08-20 00:24:15 +010052class TrcMemAccessorBase
53{
54public:
mleachc9641082015-08-24 14:38:13 +010055
mleachebc28b02015-12-11 11:15:34 +000056 /** Describes the storage type of the underlying memory accessor */
mleachc9641082015-08-24 14:38:13 +010057 enum MemAccTypes {
58 MEMACC_UNKNOWN,
mleachebc28b02015-12-11 11:15:34 +000059 MEMACC_FILE, //<! Binary data file accessor
60 MEMACC_BUFPTR, //<! memory buffer accessor
61 MEMACC_CB_IF, //<! callback interface accessor - use for live memory access
mleachc9641082015-08-24 14:38:13 +010062 };
63
mleach34d6ee12015-08-21 12:47:32 +010064 /** default constructor */
mleachc9641082015-08-24 14:38:13 +010065 TrcMemAccessorBase(MemAccTypes type);
mleach34d6ee12015-08-21 12:47:32 +010066
67 /** costruct with address range values */
mleachc9641082015-08-24 14:38:13 +010068 TrcMemAccessorBase(MemAccTypes type, rctdl_vaddr_t startAddr, rctdl_vaddr_t endAddr);
mleach34d6ee12015-08-21 12:47:32 +010069
70 /** default desctructor */
mike leacha6baf052015-08-20 00:24:15 +010071 virtual ~TrcMemAccessorBase() {};
mleach66c99f12015-08-27 17:53:11 +010072
mleach34d6ee12015-08-21 12:47:32 +010073 /*!
74 * Set the inclusive address range of this accessor.
75 *
76 * @param startAddr : start address of the range.
77 * @param endAddr : end address of the range.
78 */
mike leacha6baf052015-08-20 00:24:15 +010079 void setRange(rctdl_vaddr_t startAddr, rctdl_vaddr_t endAddr);
80
mleach34d6ee12015-08-21 12:47:32 +010081 /*!
82 * test if an address is in the inclusive range for this accessor
83 *
84 * @param s_address : Address to test.
85 *
86 * @return const bool : true if the address is in range.
87 */
mleach66c99f12015-08-27 17:53:11 +010088 virtual const bool addrInRange(const rctdl_vaddr_t s_address) const;
mleach34d6ee12015-08-21 12:47:32 +010089
mleachf1acc382015-09-14 17:27:05 +010090
91 /*!
92 * test if an address is the start of range for this accessor
93 *
94 * @param s_address : Address to test.
95 *
96 * @return const bool : true if the address is start of range.
97 */
98 virtual const bool addrStartOfRange(const rctdl_vaddr_t s_address) const;
99
mleach34d6ee12015-08-21 12:47:32 +0100100 /*!
101 * Test number of bytes available from the start address, up to the number of requested bytes.
102 * Tests if all the requested bytes are available from the supplied start address.
103 * Returns the number available up to full requested amount.
104 *
105 * @param s_address : Start address within the range.
106 * @param reqBytes : Number of bytes needed from the start address.
107 *
108 * @return const uint32_t : Bytes available, up to reqBytes. 0 is s_address not in range.
109 */
mleach66c99f12015-08-27 17:53:11 +0100110 virtual const uint32_t bytesInRange(const rctdl_vaddr_t s_address, const uint32_t reqBytes) const;
mleach34d6ee12015-08-21 12:47:32 +0100111
112 /*!
113 * test is supplied range accessor overlaps this range.
114 *
115 * @param *p_test_acc : Accessor to test for overlap.
116 *
117 * @return bool : true if overlap, false if not.
118 */
mleach66c99f12015-08-27 17:53:11 +0100119 virtual const bool overLapRange(const TrcMemAccessorBase *p_test_acc) const;
mleach34d6ee12015-08-21 12:47:32 +0100120
121 /*!
122 * Read bytes from via the accessor from the memory range.
123 *
124 * @param s_address : Start address of the read.
mleachebc28b02015-12-11 11:15:34 +0000125 * @param memSpace : memory space for this access.
mleach34d6ee12015-08-21 12:47:32 +0100126 * @param reqBytes : Number of bytes required.
127 * @param *byteBuffer : Buffer to copy the bytes into.
128 *
mleachebc28b02015-12-11 11:15:34 +0000129 * @return uint32_t : Number of bytes read, 0 if s_address out of range, or mem space not accessible.
mleach34d6ee12015-08-21 12:47:32 +0100130 */
mleachebc28b02015-12-11 11:15:34 +0000131 virtual const uint32_t readBytes(const rctdl_vaddr_t s_address, const rctdl_mem_space_acc_t memSpace, const uint32_t reqBytes, uint8_t *byteBuffer) = 0;
132
133 /*!
134 * Validate the address range - ensure addresses aligned, different, st < en etc.
135 *
136 * @return bool : true if valid range.
137 */
Mike Leach000073b2016-02-18 01:27:23 +0000138 virtual const bool validateRange();
mleachebc28b02015-12-11 11:15:34 +0000139
mike leacha6baf052015-08-20 00:24:15 +0100140
mleachc9641082015-08-24 14:38:13 +0100141 const enum MemAccTypes getType() const { return m_type; };
142
mleach66c99f12015-08-27 17:53:11 +0100143 /* handle memory spaces */
144 void setMemSpace(rctdl_mem_space_acc_t memSpace) { m_mem_space = memSpace; };
145 const rctdl_mem_space_acc_t getMemSpace() const { return m_mem_space; };
mleach5e0cde72015-10-27 14:16:20 +0000146 const bool inMemSpace(const rctdl_mem_space_acc_t mem_space) const { return (bool)(((uint8_t)m_mem_space & (uint8_t)mem_space) != 0); };
mleach66c99f12015-08-27 17:53:11 +0100147
mleachd9bd2322015-12-18 18:12:48 +0000148 /* memory access info logging */
149 virtual void getMemAccString(std::string &accStr) const;
150
mike leacha6baf052015-08-20 00:24:15 +0100151protected:
mleachebc28b02015-12-11 11:15:34 +0000152 rctdl_vaddr_t m_startAddress; /**< accessible range start address */
153 rctdl_vaddr_t m_endAddress; /**< accessible range end address */
154 const MemAccTypes m_type; /**< memory accessor type */
mleach66c99f12015-08-27 17:53:11 +0100155 rctdl_mem_space_acc_t m_mem_space;
mike leacha6baf052015-08-20 00:24:15 +0100156};
157
mleachc9641082015-08-24 14:38:13 +0100158inline TrcMemAccessorBase::TrcMemAccessorBase(MemAccTypes accType, rctdl_vaddr_t startAddr, rctdl_vaddr_t endAddr) :
mike leacha6baf052015-08-20 00:24:15 +0100159 m_startAddress(startAddr),
mleachc9641082015-08-24 14:38:13 +0100160 m_endAddress(endAddr),
mleach66c99f12015-08-27 17:53:11 +0100161 m_type(accType),
162 m_mem_space(RCTDL_MEM_SPACE_ANY)
mike leacha6baf052015-08-20 00:24:15 +0100163{
164}
165
mleachc9641082015-08-24 14:38:13 +0100166inline TrcMemAccessorBase::TrcMemAccessorBase(MemAccTypes accType) :
mike leacha6baf052015-08-20 00:24:15 +0100167 m_startAddress(0),
mleachc9641082015-08-24 14:38:13 +0100168 m_endAddress(0),
mleach66c99f12015-08-27 17:53:11 +0100169 m_type(accType),
170 m_mem_space(RCTDL_MEM_SPACE_ANY)
mike leacha6baf052015-08-20 00:24:15 +0100171{
172}
173
174inline void TrcMemAccessorBase::setRange(rctdl_vaddr_t startAddr, rctdl_vaddr_t endAddr)
175{
176 m_startAddress = startAddr;
177 m_endAddress = endAddr;
178}
179
180inline const bool TrcMemAccessorBase::addrInRange(const rctdl_vaddr_t s_address) const
181{
182 return (s_address >= m_startAddress) && (s_address <= m_endAddress);
183}
184
mleachf1acc382015-09-14 17:27:05 +0100185inline const bool TrcMemAccessorBase::addrStartOfRange(const rctdl_vaddr_t s_address) const
186{
187 return (s_address == m_startAddress);
188}
189
190
mike leacha6baf052015-08-20 00:24:15 +0100191inline const uint32_t TrcMemAccessorBase::bytesInRange(const rctdl_vaddr_t s_address, const uint32_t reqBytes) const
192{
mleach5e0cde72015-10-27 14:16:20 +0000193 rctdl_vaddr_t bytesInRange = 0;
mleacha37d6432015-08-20 17:53:50 +0100194 if(addrInRange(s_address)) // start not in range, return 0.
mike leacha6baf052015-08-20 00:24:15 +0100195 {
mleacha37d6432015-08-20 17:53:50 +0100196 // bytes available till end address.
197 bytesInRange = m_endAddress - s_address + 1;
198 if(bytesInRange > reqBytes)
mike leacha6baf052015-08-20 00:24:15 +0100199 bytesInRange = reqBytes;
mike leacha6baf052015-08-20 00:24:15 +0100200 }
mleach5e0cde72015-10-27 14:16:20 +0000201 return (uint32_t)bytesInRange;
mike leacha6baf052015-08-20 00:24:15 +0100202}
203
mleach34d6ee12015-08-21 12:47:32 +0100204inline const bool TrcMemAccessorBase::overLapRange(const TrcMemAccessorBase *p_test_acc) const
205{
mleach66c99f12015-08-27 17:53:11 +0100206 if( addrInRange(p_test_acc->m_startAddress) ||
207 addrInRange(p_test_acc->m_endAddress)
mleach34d6ee12015-08-21 12:47:32 +0100208 )
209 return true;
210 return false;
211}
212
mleachebc28b02015-12-11 11:15:34 +0000213inline const bool TrcMemAccessorBase::validateRange()
214{
215 if(m_startAddress & 0x1) // at least hword aligned for thumb
216 return false;
217 if((m_endAddress + 1) & 0x1)
218 return false;
219 if(m_startAddress == m_endAddress) // zero length range.
220 return false;
221 if(m_startAddress > m_endAddress) // values bakcwards / invalid
222 return false;
223 return true;
224}
225
226
227class TrcMemAccFactory
228{
229public:
230 /** Accessor Creation */
231 static rctdl_err_t CreateBufferAccessor(TrcMemAccessorBase **pAccessor, const rctdl_vaddr_t s_address, const uint8_t *p_buffer, const uint32_t size);
232 static rctdl_err_t CreateFileAccessor(TrcMemAccessorBase **pAccessor, const std::string &pathToFile, rctdl_vaddr_t startAddr, size_t offset = 0, size_t size = 0);
233 static rctdl_err_t CreateCBAccessor(TrcMemAccessorBase **pAccessor, const rctdl_vaddr_t s_address, const rctdl_vaddr_t e_address, const rctdl_mem_space_acc_t mem_space);
234
235 /** Accessor Destruction */
236 static void DestroyAccessor(TrcMemAccessorBase *pAccessor);
237private:
238 TrcMemAccFactory() {};
239 ~TrcMemAccFactory() {};
240};
241
mike leacha6baf052015-08-20 00:24:15 +0100242#endif // ARM_TRC_MEM_ACC_BASE_H_INCLUDED
243
244/* End of File trc_mem_acc_base.h */