1 /****************************************************************************
3 (c) SYSTEC electronic GmbH, D-07973 Greiz, August-Bebel-Str. 29
4 www.systec-electronic.com
8 Description: proc fs entry with diagnostic information under Linux
12 Redistribution and use in source and binary forms, with or without
13 modification, are permitted provided that the following conditions
16 1. Redistributions of source code must retain the above copyright
17 notice, this list of conditions and the following disclaimer.
19 2. Redistributions in binary form must reproduce the above copyright
20 notice, this list of conditions and the following disclaimer in the
21 documentation and/or other materials provided with the distribution.
23 3. Neither the name of SYSTEC electronic GmbH nor the names of its
24 contributors may be used to endorse or promote products derived
25 from this software without prior written permission. For written
26 permission, please contact info@systec-electronic.com.
28 THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
29 "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
30 LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS
31 FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE
32 COPYRIGHT HOLDERS OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT,
33 INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING,
34 BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES;
35 LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER
36 CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
37 LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN
38 ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
39 POSSIBILITY OF SUCH DAMAGE.
43 If a provision of this License is or becomes illegal, invalid or
44 unenforceable in any jurisdiction, that shall not affect:
45 1. the validity or enforceability in that jurisdiction of any other
46 provision of this License; or
47 2. the validity or enforceability in other jurisdictions of that or
48 any other provision of this License.
50 -------------------------------------------------------------------------
52 $RCSfile: proc_fs.c,v $
56 $Revision: 1.13 $ $Date: 2008/11/07 13:55:56 $
63 -------------------------------------------------------------------------
67 2006/07/31 d.k.: start of implementation
69 ****************************************************************************/
71 #include "kernel/EplNmtk.h"
72 #include "user/EplNmtu.h"
74 #if (((EPL_MODULE_INTEGRATION) & (EPL_MODULE_NMT_MN)) != 0)
75 #include "user/EplNmtMnu.h"
78 #include "kernel/EplDllkCal.h"
80 //#include <linux/config.h>
81 #include <linux/module.h>
82 #include <linux/kernel.h>
83 #include <linux/init.h>
84 #include <linux/errno.h>
85 #include <linux/major.h>
87 #include <asm/uaccess.h>
88 #include <asm/atomic.h>
89 #include <linux/proc_fs.h>
90 #include <linux/spinlock.h>
92 #ifdef CONFIG_COLDFIRE
93 #include <asm/coldfire.h>
99 /***************************************************************************/
102 /* G L O B A L D E F I N I T I O N S */
105 /***************************************************************************/
107 //---------------------------------------------------------------------------
109 //---------------------------------------------------------------------------
111 #ifndef EPL_PROC_DEV_NAME
112 #define EPL_PROC_DEV_NAME "epl"
115 #ifndef DBG_TRACE_POINTS
116 #define DBG_TRACE_POINTS 23 // # of supported debug trace points
119 #ifndef DBG_TRACE_VALUES
120 #define DBG_TRACE_VALUES 24 // # of supported debug trace values (size of circular buffer)
123 //---------------------------------------------------------------------------
124 // modul global types
125 //---------------------------------------------------------------------------
127 //---------------------------------------------------------------------------
129 //---------------------------------------------------------------------------
131 #ifdef _DBG_TRACE_POINTS_
132 atomic_t aatmDbgTracePoint_l[DBG_TRACE_POINTS];
133 u32 adwDbgTraceValue_l[DBG_TRACE_VALUES];
134 u32 dwDbgTraceValueOld_l;
135 unsigned int uiDbgTraceValuePos_l;
136 spinlock_t spinlockDbgTraceValue_l;
137 unsigned long ulDbTraceValueFlags_l;
140 //---------------------------------------------------------------------------
141 // local function prototypes
142 //---------------------------------------------------------------------------
144 static int EplLinProcRead(char *pcBuffer_p, char **ppcStart_p, off_t Offset_p,
145 int nBufferSize_p, int *pEof_p, void *pData_p);
146 static int EplLinProcWrite(struct file *file, const char __user * buffer,
147 unsigned long count, void *data);
149 void TgtDbgSignalTracePoint(u8 bTracePointNumber_p);
150 void TgtDbgPostTraceValue(u32 dwTraceValue_p);
152 extern u32 EplIdentuGetRunningRequests(void);
154 //=========================================================================//
156 // P U B L I C F U N C T I O N S //
158 //=========================================================================//
160 tEplKernel EplLinProcInit(void)
162 struct proc_dir_entry *pProcDirEntry;
163 pProcDirEntry = create_proc_entry(EPL_PROC_DEV_NAME, S_IRUGO, NULL);
164 if (pProcDirEntry != NULL) {
165 pProcDirEntry->read_proc = EplLinProcRead;
166 pProcDirEntry->write_proc = EplLinProcWrite;
167 pProcDirEntry->data = NULL; // device number or something else
170 return kEplNoResource;
173 #ifdef _DBG_TRACE_POINTS_
174 // initialize spinlock and circular buffer position
175 spin_lock_init(&spinlockDbgTraceValue_l);
176 uiDbgTraceValuePos_l = 0;
177 dwDbgTraceValueOld_l = 0;
180 return kEplSuccessful;
183 tEplKernel EplLinProcFree(void)
185 remove_proc_entry(EPL_PROC_DEV_NAME, NULL);
187 return kEplSuccessful;
190 //---------------------------------------------------------------------------
191 // Target specific event signaling (FEC Tx-/Rx-Interrupt, used by Edrv)
192 //---------------------------------------------------------------------------
194 #ifdef _DBG_TRACE_POINTS_
195 void TgtDbgSignalTracePoint(u8 bTracePointNumber_p)
198 if (bTracePointNumber_p >=
199 (sizeof(aatmDbgTracePoint_l) / sizeof(aatmDbgTracePoint_l[0]))) {
203 atomic_inc(&aatmDbgTracePoint_l[bTracePointNumber_p]);
211 void TgtDbgPostTraceValue(u32 dwTraceValue_p)
214 spin_lock_irqsave(&spinlockDbgTraceValue_l, ulDbTraceValueFlags_l);
215 if (dwDbgTraceValueOld_l != dwTraceValue_p) {
216 adwDbgTraceValue_l[uiDbgTraceValuePos_l] = dwTraceValue_p;
217 uiDbgTraceValuePos_l =
218 (uiDbgTraceValuePos_l + 1) % DBG_TRACE_VALUES;
219 dwDbgTraceValueOld_l = dwTraceValue_p;
221 spin_unlock_irqrestore(&spinlockDbgTraceValue_l, ulDbTraceValueFlags_l);
228 //---------------------------------------------------------------------------
229 // Read function for PROC-FS read access
230 //---------------------------------------------------------------------------
232 static int EplLinProcRead(char *pcBuffer_p,
235 int nBufferSize_p, int *pEof_p, void *pData_p)
240 tEplDllkCalStatistics *pDllkCalStats;
245 // count calls of this function
246 #ifdef _DBG_TRACE_POINTS_
247 TgtDbgSignalTracePoint(0);
250 //---------------------------------------------------------------
251 // generate static information
252 //---------------------------------------------------------------
254 // ---- Driver information ----
255 nSize += snprintf(pcBuffer_p + nSize, nBufferSize_p - nSize,
256 "%s %s (c) 2006 %s\n",
257 EPL_PRODUCT_NAME, EPL_PRODUCT_VERSION,
258 EPL_PRODUCT_MANUFACTURER);
260 //---------------------------------------------------------------
261 // generate process information
262 //---------------------------------------------------------------
264 // ---- EPL state ----
265 nSize += snprintf(pcBuffer_p + nSize, nBufferSize_p - nSize,
266 "NMT state: 0x%04X\n",
267 (u16) EplNmtkGetNmtState());
269 EplDllkCalGetStatistics(&pDllkCalStats);
271 nSize += snprintf(pcBuffer_p + nSize, nBufferSize_p - nSize,
272 "CurAsyncTxGen=%lu CurAsyncTxNmt=%lu CurAsyncRx=%lu\nMaxAsyncTxGen=%lu MaxAsyncTxNmt=%lu MaxAsyncRx=%lu\n",
273 pDllkCalStats->m_ulCurTxFrameCountGen,
274 pDllkCalStats->m_ulCurTxFrameCountNmt,
275 pDllkCalStats->m_ulCurRxFrameCount,
276 pDllkCalStats->m_ulMaxTxFrameCountGen,
277 pDllkCalStats->m_ulMaxTxFrameCountNmt,
278 pDllkCalStats->m_ulMaxRxFrameCount);
280 #if (((EPL_MODULE_INTEGRATION) & (EPL_MODULE_NMT_MN)) != 0)
281 // fetch running IdentRequests
282 nSize += snprintf(pcBuffer_p + nSize, nBufferSize_p - nSize,
283 "running IdentRequests: 0x%08X\n",
284 EplIdentuGetRunningRequests());
286 // fetch state of NmtMnu module
288 unsigned int uiMandatorySlaveCount;
289 unsigned int uiSignalSlaveCount;
292 EplNmtMnuGetDiagnosticInfo(&uiMandatorySlaveCount,
293 &uiSignalSlaveCount, &wFlags);
295 nSize += snprintf(pcBuffer_p + nSize, nBufferSize_p - nSize,
296 "MN MandSlaveCount: %u SigSlaveCount: %u Flags: 0x%X\n",
297 uiMandatorySlaveCount, uiSignalSlaveCount,
303 // ---- FEC state ----
304 #ifdef CONFIG_COLDFIRE
306 // Receive the base address
307 unsigned long base_addr;
308 #if (EDRV_USED_ETH_CTRL == 0)
309 // Set the base address of FEC0
310 base_addr = FEC_BASE_ADDR_FEC0;
312 // Set the base address of FEC1
313 base_addr = FEC_BASE_ADDR_FEC1;
316 nSize += snprintf(pcBuffer_p + nSize, nBufferSize_p - nSize,
317 "FEC_ECR = 0x%08X FEC_EIR = 0x%08X FEC_EIMR = 0x%08X\nFEC_TCR = 0x%08X FECTFSR = 0x%08X FECRFSR = 0x%08X\n",
318 FEC_ECR(base_addr), FEC_EIR(base_addr),
319 FEC_EIMR(base_addr), FEC_TCR(base_addr),
320 FEC_FECTFSR(base_addr),
321 FEC_FECRFSR(base_addr));
325 // ---- DBG: TracePoints ----
326 #ifdef _DBG_TRACE_POINTS_
330 nSize += snprintf(pcBuffer_p + nSize, nBufferSize_p - nSize,
331 "DbgTracePoints:\n");
333 nNum < (sizeof(aatmDbgTracePoint_l) / sizeof(atomic_t));
336 snprintf(pcBuffer_p + nSize, nBufferSize_p - nSize,
337 " TracePoint[%2d]: %d\n", (int)nNum,
338 atomic_read(&aatmDbgTracePoint_l[nNum]));
341 nSize += snprintf(pcBuffer_p + nSize, nBufferSize_p - nSize,
342 "DbgTraceValues:\n");
343 for (nNum = 0; nNum < DBG_TRACE_VALUES; nNum++) {
344 if (nNum == uiDbgTraceValuePos_l) { // next value will be stored at that position
346 snprintf(pcBuffer_p + nSize,
347 nBufferSize_p - nSize, "*%08lX",
348 adwDbgTraceValue_l[nNum]);
351 snprintf(pcBuffer_p + nSize,
352 nBufferSize_p - nSize, " %08lX",
353 adwDbgTraceValue_l[nNum]);
355 if ((nNum & 0x00000007) == 0x00000007) { // 8 values printed -> end of line reached
357 snprintf(pcBuffer_p + nSize,
358 nBufferSize_p - nSize, "\n");
361 if ((nNum & 0x00000007) != 0x00000007) { // number of values printed is not a multiple of 8 -> print new line
363 snprintf(pcBuffer_p + nSize, nBufferSize_p - nSize,
380 //---------------------------------------------------------------------------
381 // Write function for PROC-FS write access
382 //---------------------------------------------------------------------------
384 static int EplLinProcWrite(struct file *file, const char __user * buffer,
385 unsigned long count, void *data)
387 char abBuffer[count + 1];
390 tEplNmtEvent NmtEvent;
393 iErr = copy_from_user(abBuffer, buffer, count);
397 abBuffer[count] = '\0';
399 iErr = sscanf(abBuffer, "%i", &iVal);
401 if ((iVal <= 0) || (iVal > 0x2F)) {
402 NmtEvent = kEplNmtEventSwReset;
404 NmtEvent = (tEplNmtEvent) iVal;
406 // execute specified NMT command on write access of /proc/epl
407 EplNmtuNmtEvent(NmtEvent);