/**************************************************************************** * * Copyright (C) 2005 - 2013 by Vivante Corp. * * This program is free software; you can redistribute it and/or modify * it under the terms of the GNU General Public License as published by * the Free Software Foundation; either version 2 of the license, or * (at your option) any later version. * * This program is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the * GNU General Public License for more details. * * You should have received a copy of the GNU General Public License * along with this program; if not write to the Free Software * Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA. * *****************************************************************************/ #ifdef MODULE #include #endif #include #include #include #ifdef MODVERSIONS #include #endif #include #include #include #include #include #include #include #include #include #include #include #include #include #include "gc_hal_kernel_linux.h" /* Prequsite: 1) Debugfs feature must be enabled in the kernel. 1.a) You can enable this, in the compilation of the uImage, all you have to do is, In the "make menuconfig" part, you have to enable the debugfs in the kernel hacking part of the menu. HOW TO USE: 1) insert the driver with the following option logFileSize, Ex: insmod galcore.ko ...... logFileSize=10240 This gives a circular buffer of 10 MB 2)Usually after inserting the driver, the debug file system is mounted under /sys/kernel/debug/ 2.a)If the debugfs is not mounted, you must do "mount -t debugfs none /sys/kernel/debug" 3) To read what is being printed in the debugfs file system: Ex : cat /sys/kernel/debug/gpu/galcore_trace 4)To write into the debug file system from user side : Ex: echo "hello" > cat /sys/kernel/debug/gpu/galcore_trace 5)To write into debugfs from kernel side, Use the function called gckDebugFileSystemPrint USECASE Kernel Dump: 1) Go to /hal/inc/gc_hal_options.h, and enable the following flags: - # define gcdDUMP 1 - # define gcdDUMP_IN_KERNEL 1 - # define gcdDUMP_COMMAND 1 2) Go to /hal/kernel/gc_hal_kernel_command.c and disable the following flag -#define gcdSIMPLE_COMMAND_DUMP 0 3) Compile the driver 4) insmod it with the logFileSize option 5) Run an application 6) You can get the dump by cat /sys/kernel/debug/gpu/galcore_trace */ /**/ typedef va_list gctDBGARGS ; #define gcmkARGS_START(argument, pointer) va_start(argument, pointer) #define gcmkARGS_END(argument) va_end(argument) #define gcmkDBGFSPRINT(ArgumentSize, Message) \ { \ gctDBGARGS __arguments__; \ gcmkARGS_START(__arguments__, Message); \ _DebugFSPrint(ArgumentSize, Message, __arguments__);\ gcmkARGS_END(__arguments__); \ } /*Debug File System Node Struct*/ struct _gcsDebugFileSystemNode { /*wait queues for read and write operations*/ #if defined(DECLARE_WAIT_QUEUE_HEAD) wait_queue_head_t read_q , write_q ; #else struct wait_queue *read_q , *write_q ; #endif struct dentry *parent ; /*parent directory*/ struct dentry *filen ; /*filename*/ struct semaphore sem ; /* mutual exclusion semaphore */ char *data ; /* The circular buffer data */ int size ; /* Size of the buffer pointed to by 'data' */ int refcount ; /* Files that have this buffer open */ int read_point ; /* Offset in circ. buffer of oldest data */ int write_point ; /* Offset in circ. buffer of newest data */ int offset ; /* Byte number of read_point in the stream */ struct _gcsDebugFileSystemNode *next ; } ; /* amount of data in the queue */ #define gcmkNODE_QLEN(node) ( (node)->write_point >= (node)->read_point ? \ (node)->write_point - (node)->read_point : \ (node)->size - (node)->read_point + (node)->write_point) /* byte number of the last byte in the queue */ #define gcmkNODE_FIRST_EMPTY_BYTE(node) ((node)->offset + gcmkNODE_QLEN(node)) /*Synchronization primitives*/ #define gcmkNODE_READQ(node) (&((node)->read_q)) #define gcmkNODE_WRITEQ(node) (&((node)->write_q)) #define gcmkNODE_SEM(node) (&((node)->sem)) /*Utilities*/ #define gcmkMIN(x, y) ((x) < (y) ? (x) : y) /*Debug File System Struct*/ typedef struct _gcsDebugFileSystem { gcsDebugFileSystemNode* linkedlist ; gcsDebugFileSystemNode* currentNode ; int isInited ; } gcsDebugFileSystem ; /*debug file system*/ static gcsDebugFileSystem gc_dbgfs ; /******************************************************************************* ** ** READ & WRITE FUNCTIONS (START) ** *******************************************************************************/ /******************************************************************************* ** ** _ReadFromNode ** ** 1) reading bytes out of a circular buffer with wraparound. ** 2)returns caddr_t, pointer to data read, which the caller must free. ** 3) length is (a pointer to) the number of bytes to be read, which will be set by this function to ** be the number of bytes actually returned ** *******************************************************************************/ static caddr_t _ReadFromNode ( gcsDebugFileSystemNode* Node , size_t *Length , loff_t *Offset ) { caddr_t retval ; int bytes_copied = 0 , n , start_point , remaining ; /* is the user trying to read data that has already scrolled off? */ if ( *Offset < Node->offset ) { *Offset = Node->offset ; } /* is the user trying to read past EOF? */ if ( *Offset >= gcmkNODE_FIRST_EMPTY_BYTE ( Node ) ) { return NULL ; } /* find the smaller of the total bytes we have available and what * the user is asking for */ *Length = gcmkMIN ( *Length , gcmkNODE_FIRST_EMPTY_BYTE ( Node ) - *Offset ) ; remaining = * Length ; /* figure out where to start based on user's Offset */ start_point = Node->read_point + ( *Offset - Node->offset ) ; start_point = start_point % Node->size ; /* allocate memory to return */ if ( ( retval = kmalloc ( sizeof (char ) * remaining , GFP_KERNEL ) ) == NULL ) return NULL ; /* copy the (possibly noncontiguous) data to our buffer */ while ( remaining ) { n = gcmkMIN ( remaining , Node->size - start_point ) ; memcpy ( retval + bytes_copied , Node->data + start_point , n ) ; bytes_copied += n ; remaining -= n ; start_point = ( start_point + n ) % Node->size ; } /* advance user's file pointer */ *Offset += * Length ; return retval ; } /******************************************************************************* ** ** _WriteToNode ** ** 1) writes to a circular buffer with wraparound. ** 2)in case of an overflow, it overwrites the oldest unread data. ** *********************************************************************************/ static void _WriteToNode ( gcsDebugFileSystemNode* Node , caddr_t Buf , int Length ) { int bytes_copied = 0 ; int overflow = 0 ; int n ; if ( Length + gcmkNODE_QLEN ( Node ) >= ( Node->size - 1 ) ) { overflow = 1 ; /* in case of overflow, figure out where the new buffer will * begin. we start by figuring out where the current buffer ENDS: * node->parent->offset + gcmkNODE_QLEN. we then advance the end-offset * by the Length of the current write, and work backwards to * figure out what the oldest unoverwritten data will be (i.e., * size of the buffer). */ Node->offset = Node->offset + gcmkNODE_QLEN ( Node ) + Length - Node->size + 1 ; } while ( Length ) { /* how many contiguous bytes are available from the write point to * the end of the circular buffer? */ n = gcmkMIN ( Length , Node->size - Node->write_point ) ; memcpy ( Node->data + Node->write_point , Buf + bytes_copied , n ) ; bytes_copied += n ; Length -= n ; Node->write_point = ( Node->write_point + n ) % Node->size ; } /* if there is an overflow, reset the read point to read whatever is * the oldest data that we have, that has not yet been * overwritten. */ if ( overflow ) { Node->read_point = ( Node->write_point + 1 ) % Node->size ; } } /******************************************************************************* ** ** PRINTING UTILITY (START) ** *******************************************************************************/ /******************************************************************************* ** ** _GetArgumentSize ** ** *******************************************************************************/ static gctINT _GetArgumentSize ( IN gctCONST_STRING Message ) { gctINT i , count ; for ( i = 0 , count = 0 ; Message[i] ; i += 1 ) { if ( Message[i] == '%' ) { count += 1 ; } } return count * sizeof (unsigned int ) ; } /******************************************************************************* ** ** _AppendString ** ** *******************************************************************************/ static ssize_t _AppendString ( IN gcsDebugFileSystemNode* Node , IN gctCONST_STRING String , IN int Length ) { caddr_t message = NULL ; int n ; /* if the message is longer than the buffer, just take the beginning * of it, in hopes that the reader (if any) will have time to read * before we wrap around and obliterate it */ n = gcmkMIN ( Length , Node->size - 1 ) ; /* make sure we have the memory for it */ if ( ( message = kmalloc ( n , GFP_KERNEL ) ) == NULL ) return - ENOMEM ; /* copy into our temp buffer */ memcpy ( message , String , n ) ; /* now copy it into the circular buffer and free our temp copy */ _WriteToNode ( Node , message , n ) ; kfree ( message ) ; return n ; } /******************************************************************************* ** ** _DebugFSPrint ** ** *******************************************************************************/ static void _DebugFSPrint ( IN unsigned int ArgumentSize , IN const char* Message , IN gctDBGARGS Arguments ) { char buffer[MAX_LINE_SIZE] ; int len ; down ( gcmkNODE_SEM ( gc_dbgfs.currentNode ) ) ; len = vsnprintf ( buffer , sizeof (buffer ) , Message , *( va_list * ) & Arguments ) ; buffer[len] = '\0' ; /* Add end-of-line if missing. */ if ( buffer[len - 1] != '\n' ) { buffer[len ++] = '\n' ; buffer[len] = '\0' ; } _AppendString ( gc_dbgfs.currentNode , buffer , len ) ; up ( gcmkNODE_SEM ( gc_dbgfs.currentNode ) ) ; wake_up_interruptible ( gcmkNODE_READQ ( gc_dbgfs.currentNode ) ) ; /* blocked in read*/ } /******************************************************************************* ** ** LINUX SYSTEM FUNCTIONS (START) ** *******************************************************************************/ /******************************************************************************* ** ** find the vivlog structure associated with an inode. ** returns a pointer to the structure if found, NULL if not found ** *******************************************************************************/ static gcsDebugFileSystemNode* _GetNodeInfo ( IN struct inode *Inode ) { gcsDebugFileSystemNode* node ; if ( Inode == NULL ) return NULL ; for ( node = gc_dbgfs.linkedlist ; node != NULL ; node = node->next ) if ( node->filen->d_inode->i_ino == Inode->i_ino ) return node ; return NULL ; } /******************************************************************************* ** ** _DebugFSRead ** *******************************************************************************/ static ssize_t _DebugFSRead ( struct file *file , char __user * buffer , size_t length , loff_t * offset ) { int retval ; caddr_t data_to_return ; gcsDebugFileSystemNode* node ; /* get the metadata about this emlog */ if ( ( node = _GetNodeInfo ( file->f_dentry->d_inode ) ) == NULL ) { printk ( "debugfs_read: record not found\n" ) ; return - EIO ; } if ( down_interruptible ( gcmkNODE_SEM ( node ) ) ) { return - ERESTARTSYS ; } /* wait until there's data available (unless we do nonblocking reads) */ while ( *offset >= gcmkNODE_FIRST_EMPTY_BYTE ( node ) ) { up ( gcmkNODE_SEM ( node ) ) ; if ( file->f_flags & O_NONBLOCK ) { return - EAGAIN ; } if ( wait_event_interruptible ( ( *( gcmkNODE_READQ ( node ) ) ) , ( *offset < gcmkNODE_FIRST_EMPTY_BYTE ( node ) ) ) ) { return - ERESTARTSYS ; /* signal: tell the fs layer to handle it */ } /* otherwise loop, but first reacquire the lock */ if ( down_interruptible ( gcmkNODE_SEM ( node ) ) ) { return - ERESTARTSYS ; } } data_to_return = _ReadFromNode ( node , &length , offset ) ; if ( data_to_return == NULL ) { retval = 0 ; goto unlock ; } if ( copy_to_user ( buffer , data_to_return , length ) > 0 ) { retval = - EFAULT ; } else { retval = length ; } kfree ( data_to_return ) ; unlock: up ( gcmkNODE_SEM ( node ) ) ; wake_up_interruptible ( gcmkNODE_WRITEQ ( node ) ) ; return retval ; } /******************************************************************************* ** **_DebugFSWrite ** *******************************************************************************/ static ssize_t _DebugFSWrite ( struct file *file , const char __user * buffer , size_t length , loff_t * offset ) { caddr_t message = NULL ; int n ; gcsDebugFileSystemNode*node ; /* get the metadata about this log */ if ( ( node = _GetNodeInfo ( file->f_dentry->d_inode ) ) == NULL ) { return - EIO ; } if ( down_interruptible ( gcmkNODE_SEM ( node ) ) ) { return - ERESTARTSYS ; } /* if the message is longer than the buffer, just take the beginning * of it, in hopes that the reader (if any) will have time to read * before we wrap around and obliterate it */ n = gcmkMIN ( length , node->size - 1 ) ; /* make sure we have the memory for it */ if ( ( message = kmalloc ( n , GFP_KERNEL ) ) == NULL ) { up ( gcmkNODE_SEM ( node ) ) ; return - ENOMEM ; } /* copy into our temp buffer */ if ( copy_from_user ( message , buffer , n ) > 0 ) { up ( gcmkNODE_SEM ( node ) ) ; kfree ( message ) ; return - EFAULT ; } /* now copy it into the circular buffer and free our temp copy */ _WriteToNode ( node , message , n ) ; kfree ( message ) ; up ( gcmkNODE_SEM ( node ) ) ; /* wake up any readers that might be waiting for the data. we call * schedule in the vague hope that a reader will run before the * writer's next write, to avoid losing data. */ wake_up_interruptible ( gcmkNODE_READQ ( node ) ) ; return n ; } /******************************************************************************* ** ** File Operations Table ** *******************************************************************************/ static const struct file_operations debugfs_operations = { .owner = THIS_MODULE , .read = _DebugFSRead , .write = _DebugFSWrite , } ; /******************************************************************************* ** ** INTERFACE FUNCTIONS (START) ** *******************************************************************************/ /******************************************************************************* ** ** gckDebugFileSystemIsEnabled ** ** ** INPUT: ** ** OUTPUT: ** *******************************************************************************/ gctINT gckDebugFileSystemIsEnabled ( void ) { return gc_dbgfs.isInited ; } /******************************************************************************* ** ** gckDebugFileSystemInitialize ** ** ** INPUT: ** ** OUTPUT: ** *******************************************************************************/ gctINT gckDebugFileSystemInitialize ( void ) { if ( ! gc_dbgfs.isInited ) { gc_dbgfs.linkedlist = gcvNULL ; gc_dbgfs.currentNode = gcvNULL ; gc_dbgfs.isInited = 1 ; } return gc_dbgfs.isInited ; } /******************************************************************************* ** ** gckDebugFileSystemTerminate ** ** ** INPUT: ** ** OUTPUT: ** *******************************************************************************/ gctINT gckDebugFileSystemTerminate ( void ) { gcsDebugFileSystemNode * next = gcvNULL ; gcsDebugFileSystemNode * temp = gcvNULL ; if ( gc_dbgfs.isInited ) { temp = gc_dbgfs.linkedlist ; while ( temp != gcvNULL ) { next = temp->next ; gckDebugFileSystemFreeNode ( temp ) ; kfree ( temp ) ; temp = next ; } gc_dbgfs.isInited = 0 ; } return 0 ; } /******************************************************************************* ** ** gckDebugFileSystemCreateNode ** ** ** INPUT: ** ** OUTPUT: ** ** gckDebugFileSystemFreeNode * Device ** Pointer to a variable receiving the gcsDebugFileSystemNode object pointer on ** success. *********************************************************************************/ gctINT gckDebugFileSystemCreateNode ( IN gctINT SizeInKB , IN gctCONST_STRING ParentName , IN gctCONST_STRING NodeName , OUT gcsDebugFileSystemNode **Node ) { gcsDebugFileSystemNode*node ; /* allocate space for our metadata and initialize it */ if ( ( node = kmalloc ( sizeof (gcsDebugFileSystemNode ) , GFP_KERNEL ) ) == NULL ) goto struct_malloc_failed ; /*Zero it out*/ memset ( node , 0 , sizeof (gcsDebugFileSystemNode ) ) ; /*Init the sync primitives*/ #if defined(DECLARE_WAIT_QUEUE_HEAD) init_waitqueue_head ( gcmkNODE_READQ ( node ) ) ; #else init_waitqueue ( gcmkNODE_READQ ( node ) ) ; #endif #if defined(DECLARE_WAIT_QUEUE_HEAD) init_waitqueue_head ( gcmkNODE_WRITEQ ( node ) ) ; #else init_waitqueue ( gcmkNODE_WRITEQ ( node ) ) ; #endif sema_init ( gcmkNODE_SEM ( node ) , 1 ) ; /*End the sync primitives*/ /* figure out how much of a buffer this should be and allocate the buffer */ node->size = 1024 * SizeInKB ; if ( ( node->data = ( char * ) vmalloc ( sizeof (char ) * node->size ) ) == NULL ) goto data_malloc_failed ; /*creating the debug file system*/ node->parent = debugfs_create_dir ( ParentName , NULL ) ; /*creating the file*/ node->filen = debugfs_create_file ( NodeName , S_IRUGO | S_IWUSR , node->parent , NULL , &debugfs_operations ) ; /* add it to our linked list */ node->next = gc_dbgfs.linkedlist ; gc_dbgfs.linkedlist = node ; /* pass the struct back */ *Node = node ; return 0 ; vfree ( node->data ) ; data_malloc_failed: kfree ( node ) ; struct_malloc_failed: return - ENOMEM ; } /******************************************************************************* ** ** gckDebugFileSystemFreeNode ** ** ** INPUT: ** ** OUTPUT: ** *******************************************************************************/ void gckDebugFileSystemFreeNode ( IN gcsDebugFileSystemNode * Node ) { gcsDebugFileSystemNode **ptr ; if ( Node == NULL ) { printk ( "null passed to free_vinfo\n" ) ; return ; } down ( gcmkNODE_SEM ( Node ) ) ; /*free data*/ vfree ( Node->data ) ; /*Close Debug fs*/ if ( Node->filen ) { debugfs_remove ( Node->filen ) ; } if ( Node->parent ) { debugfs_remove ( Node->parent ) ; } /* now delete the node from the linked list */ ptr = & ( gc_dbgfs.linkedlist ) ; while ( *ptr != Node ) { if ( ! *ptr ) { printk ( "corrupt info list!\n" ) ; break ; } else ptr = & ( ( **ptr ).next ) ; } *ptr = Node->next ; up ( gcmkNODE_SEM ( Node ) ) ; } /******************************************************************************* ** ** gckDebugFileSystemSetCurrentNode ** ** ** INPUT: ** ** OUTPUT: ** *******************************************************************************/ void gckDebugFileSystemSetCurrentNode ( IN gcsDebugFileSystemNode * Node ) { gc_dbgfs.currentNode = Node ; } /******************************************************************************* ** ** gckDebugFileSystemGetCurrentNode ** ** ** INPUT: ** ** OUTPUT: ** *******************************************************************************/ void gckDebugFileSystemGetCurrentNode ( OUT gcsDebugFileSystemNode ** Node ) { *Node = gc_dbgfs.currentNode ; } /******************************************************************************* ** ** gckDebugFileSystemPrint ** ** ** INPUT: ** ** OUTPUT: ** *******************************************************************************/ void gckDebugFileSystemPrint ( IN gctCONST_STRING Message , ... ) { gcmkDBGFSPRINT ( _GetArgumentSize ( Message ) , Message ) ; }