While I am freeing the memory it it causing glibc detected error. It works fine when I give .(current directory containing single file as input). However ..(parent directory) input gives this error. I have tried to use gdb with no success. Can someone please point out the problem. Thanks. Below is the code.
/*
*********************************************************************
*File Name - function.c
Purpose - Implementation of functions like iteration, open_read to
printf directories and file properties.
**********************************************************************
*/
#include "header.h"
gboolean iterator(GNode* _node, gpointer data)
{
guint _node_depth = g_node_depth(_node);
printf("%p insert _ \t%p data\n",_node,_node->data);
if( _node_depth>1 )
{
printf("%.*s",(_node_depth-2)*2," ");
printf("%s\n",(char *)_node->data);
}
return SUCCESS;
}
gboolean destroyer(GNode* node, gpointer data)
{
printf("\nfrom destroyer: ");
printf("%pnode\t data %p\n",node,node->data);
free(node->data);
return SUCCESS;
}
void open_read( char* input_dir ,GNode* parent)
{
int count = 0;
DIR *dir;
struct dirent *dp;
char *stat_path = NULL;
struct stat file_stat;
char *data = NULL;
char *formatted_data = NULL;
if(NULL == (dir = opendir( input_dir ) ) )
{
printf("cannot open %s directory",input_dir);
}
else
{
/* Loop through directory entries. */
while( (dp = readdir(dir) ) != NULL )
{
if( count >= 2 ) /* omitting '.' and '..' files */
{
if( dp->d_type == DT_REG )
{
/* Get entry's information. */
stat_path = ( char * )malloc( sizeof( char ) * MAX_SIZE );
strcpy( stat_path, input_dir );
strcat( stat_path, dp->d_name );
if ( stat( stat_path, &file_stat) == -1 )
{
printf("cant read the stats of %s",dp->d_name);
}
else
{
data=(char*)malloc(sizeof(char) * MAX_SIZE);
strcpy(data,dp->d_name);
strcat(data,"\n\tLinks\tUid\ti_node\tSize\tPath\n");
formatted_data=(char*)malloc(sizeof(char) * MAX_SIZE);
sprintf(formatted_data,"\t%d\t%d\t%d\t%d\t%s", (int)file_stat.st_nlink, (int)file_stat.st_uid, (int)file_stat.st_ino, (int)file_stat.st_size,input_dir);
strcat(data,formatted_data);
//free(formatted_data);
g_node_insert(parent,-1,g_node_new(data));
}
free(stat_path);
}
else if( dp->d_type == DT_DIR )
{
char *sub_dir = (char *)malloc(sizeof(char) * MAX_SIZE);
strcpy( sub_dir, input_dir );
strcat( sub_dir, dp->d_name);
strcat( sub_dir, "/");
open_read(sub_dir, g_node_insert(parent,-1,g_node_new(dp->d_name)) );
}
}
count++;
}
}
closedir(dir);
}
/**********************************************************************
Purpose - Simulating the LS command
Input - directory path from the command line
Output - Displaying the list of directories and file properties
return - SUCCESS OR FAILURE
***********************************************************************/
#include "header.h"
int main(int argc, char* argv[])
{
char *user_dir = NULL;
/*allocating memory*/
user_dir = (char*)malloc(sizeof(char) * MAX_SIZE);
if (NULL == user_dir)
{
fprintf(stderr, "cant allocate memory...");
return FAILURE;
}
/*opening and reading the directory by calling open_read()*/
if (argc < 2)
{
strcpy(user_dir,"./");
}
else
{
strcpy(user_dir, argv[1]);
strcat(user_dir, "/");
}
GNode * root = g_node_new(user_dir);
//g_node_insert(root);
open_read(user_dir, root);
g_node_traverse(root, G_PRE_ORDER, G_TRAVERSE_ALL, -1, (GNodeTraverseFunc)iterator, NULL);
//printf("from main():%s",root->data);
g_node_traverse(root, G_PRE_ORDER, G_TRAVERSE_ALL, -1, (GNodeTraverseFunc)destroyer, NULL);
g_node_destroy(root);
return SUCCESS;
}
char *sub_dir = (char *)malloc(sizeof(char) * MAX_SIZE);
strcpy( sub_dir, input_dir );
strcat( sub_dir, dp->d_name);
strcat( sub_dir, "/");
open_read(sub_dir, g_node_insert(parent,-1,g_node_new(dp->d_name)) );
Okay, so what you pass to g_node_new
is not what malloc
returned, so you certainly can't pass that to free
.
gboolean destroyer(GNode* node, gpointer data)
{
printf("\nfrom destroyer: ");
printf("%pnode\t data %p\n",node,node->data);
free(node->data);
return SUCCESS;
}
But you do!
I believe you intended:
open_read(sub_dir, g_node_insert(parent,-1,g_node_new(sub_dir)) );