/* This FileX test concentrates on the Fault-Tolerant log recover directory operation. */ /* For FAT 12, 16, 32, one cluster size is 1024 bytes; 1024 1024 1024 1024 1024 1024 |------------|------------|-----------|------------|------------|------------| | TEST.TXT | TEST1.TXT | TEST2.TXT | | | | |------------|------------|-----------|------------|------------|------------| 28 1000 19 Check recover directory operation: Step1: Format and open the media; Step2: Enable fault tolerant feature; Step3: Write 28 bytes into TEST.TXT; Step4: Write 1000 bytes into TEST1.TXT; Step5: Write 19 bytes into TEST2.TXT; Step6: Create new thread to continue to write 2048 data into TEST1.TXT(greater than one cluster): Step7: Terminate the new thread to simulate poweroff when update the directory into media and the redo log has been generated. Step8: Create new thread and enable fault tolerant feature to recover the data(redo operation); Step9: Terminate the new thread to simulate poweroff when recover the directory; Step10: Open the media and enable fault tolerant feature to recover the data(redo operation); Step11: Check the test files. */ #ifndef FX_STANDALONE_ENABLE #include "tx_api.h" #else #define _GNU_SOURCE #define _DEFAULT_SOURCE #include #include #endif #include "fx_api.h" #include "fx_system.h" #include "fx_fault_tolerant.h" #include #include "fx_ram_driver_test.h" extern void test_control_return(UINT status); void filex_fault_tolerant_log_recover_directory_test_application_define(void *first_unused_memory); #if defined (FX_ENABLE_FAULT_TOLERANT) && defined (FX_FAULT_TOLERANT) && defined (FX_FAULT_TOLERANT_DATA) #define DEMO_STACK_SIZE 4096 #define CACHE_SIZE 2048 #define FAULT_TOLERANT_SIZE FX_FAULT_TOLERANT_MINIMAL_BUFFER_SIZE /* Define the ThreadX and FileX object control blocks... */ #ifndef FX_STANDALONE_ENABLE static TX_THREAD ftest_0; static TX_THREAD ftest_1; static TX_THREAD ftest_2; #else static pthread_t ptid1; static pthread_t ptid2; #endif static FX_MEDIA ram_disk; static FX_FILE my_file; static UCHAR *pointer; /* Define the counters used in the test application... */ #ifndef FX_STANDALONE_ENABLE static UCHAR *cache_buffer; static UCHAR *fault_tolerant_buffer; static UCHAR *thread_1_buffer; static UCHAR *thread_2_buffer; #else static UCHAR cache_buffer[CACHE_SIZE]; static UCHAR fault_tolerant_buffer[FAULT_TOLERANT_SIZE]; #endif static UINT error_couter = 0; static UINT directory_write_interrupt_1 = FX_FALSE; static UINT directory_write_interrupt_2 = FX_FALSE; static CHAR write_buffer[4096]; static UINT write_buffer_size = 4096; static UINT write_index; static CHAR read_buffer[4096]; static UINT read_buffer_size = 4096; static UINT i; #define TEST_COUNT 3 /* FAT12, 16, 32. */ /* Define thread prototypes. */ static void ftest_0_entry(ULONG thread_input); #ifndef FX_STANDALONE_ENABLE static void ftest_1_entry(ULONG thread_input); static void ftest_2_entry(ULONG thread_input); #else static void * ftest_1_entry(void * thread_input); static void * ftest_2_entry(void * thread_input); #endif extern void _fx_ram_driver(FX_MEDIA *media_ptr); extern void test_control_return(UINT status); extern UINT _filex_fault_tolerant_log_check(FX_MEDIA *media_ptr); extern UINT (*driver_write_callback)(FX_MEDIA *media_ptr, UINT sector_type, UCHAR *block_ptr, UINT *operation_ptr); static UINT my_driver_write_1(FX_MEDIA *media_ptr, UINT sector_type, UCHAR *block_ptr, UINT *operation_ptr); static UINT my_driver_write_2(FX_MEDIA *media_ptr, UINT sector_type, UCHAR *block_ptr, UINT *operation_ptr); /* Define what the initial system looks like. */ #ifdef CTEST void test_application_define(void *first_unused_memory) #else void filex_fault_tolerant_log_recover_directory_test_application_define(void *first_unused_memory) #endif { #ifndef FX_STANDALONE_ENABLE /* Setup the working pointer. */ pointer = (UCHAR *) first_unused_memory; /* Create the main thread. */ tx_thread_create(&ftest_0, "thread 0", ftest_0_entry, 0, pointer, DEMO_STACK_SIZE, 4, 4, TX_NO_TIME_SLICE, TX_AUTO_START); pointer = pointer + DEMO_STACK_SIZE; /* Setup memory for the RAM disk and the sector cache. */ cache_buffer = pointer; pointer += CACHE_SIZE; fault_tolerant_buffer = pointer; pointer += FAULT_TOLERANT_SIZE; thread_1_buffer = pointer; pointer += DEMO_STACK_SIZE; thread_2_buffer = pointer; pointer += DEMO_STACK_SIZE; #endif /* Initialize the FileX system. */ fx_system_initialize(); #ifdef FX_STANDALONE_ENABLE ftest_0_entry(0); #endif } /* Define the test threads. */ static void ftest_0_entry(ULONG thread_input) { UINT status; ULONG actual; FX_PARAMETER_NOT_USED(thread_input); /* Print out some test information banners. */ printf("FileX Test: Fault Tolerant Log Recover DIR Test...................."); /* Random genearte the write data. */ for (write_index = 0; write_index < write_buffer_size; write_index ++) write_buffer[write_index] = (CHAR)rand(); /* Loop to test FAT 12, 16, 32. */ for (i = 0; i < TEST_COUNT; i ++) { if (i == 0) { /* Format the media with FAT12. This needs to be done before opening it! */ status = fx_media_format(&ram_disk, _fx_ram_driver, // Driver entry ram_disk_memory_large, // RAM disk memory pointer cache_buffer, // Media buffer pointer CACHE_SIZE, // Media buffer size "MY_RAM_DISK", // Volume Name 1, // Number of FATs 32, // Directory Entries 0, // Hidden sectors 256, // Total sectors 256, // Sector size 8, // Sectors per cluster 1, // Heads 1); // Sectors per track } else if (i == 1) { /* Format the media with FAT16. This needs to be done before opening it! */ status = fx_media_format(&ram_disk, _fx_ram_driver, // Driver entry ram_disk_memory_large, // RAM disk memory pointer cache_buffer, // Media buffer pointer CACHE_SIZE, // Media buffer size "MY_RAM_DISK", // Volume Name 1, // Number of FATs 32, // Directory Entries 0, // Hidden sectors 4200 * 8, // Total sectors 256, // Sector size 8, // Sectors per cluster 1, // Heads 1); // Sectors per track } else if (i == 2) { /* Format the media with FAT32. This needs to be done before opening it! */ status = fx_media_format(&ram_disk, _fx_ram_driver, // Driver entry ram_disk_memory_large, // RAM disk memory pointer cache_buffer, // Media buffer pointer CACHE_SIZE, // Media buffer size "MY_RAM_DISK", // Volume Name 1, // Number of FATs 32, // Directory Entries 0, // Hidden sectors 70000 * 8, // Total sectors 256, // Sector size 8, // Sectors per cluster 1, // Heads 1); // Sectors per track } /* Determine if the format had an error. */ if (status) { printf("ERROR!\n"); test_control_return(1); } /* Open the ram_disk. */ status = fx_media_open(&ram_disk, "RAM DISK", _fx_ram_driver, ram_disk_memory_large, cache_buffer, CACHE_SIZE); /* Check the status. */ if (status != FX_SUCCESS) { /* Error, return error code. */ printf("ERROR!\n"); test_control_return(2); } /* Enable the Fault-tolerant feature. */ status = fx_fault_tolerant_enable(&ram_disk, fault_tolerant_buffer, FAULT_TOLERANT_SIZE); /* Check status. */ if (status) { printf("ERROR!\n"); test_control_return(3); } /* Create a file called TEST.TXT in the root directory. */ status = fx_file_create(&ram_disk, "TEST.TXT"); /* Check the create status. */ if (status != FX_SUCCESS) { printf("ERROR!\n"); test_control_return(4); } /* Open the test file. */ status = fx_file_open(&ram_disk, &my_file, "TEST.TXT", FX_OPEN_FOR_WRITE); /* Check the file open status. */ if (status != FX_SUCCESS) { printf("ERROR!\n"); test_control_return(5); } /* Write a string to the test file. */ status = fx_file_write(&my_file, " ABCDEFGHIJKLMNOPQRSTUVWXYZ\n", 28); /* Check the file write status. */ if (status != FX_SUCCESS) { printf("ERROR!\n"); test_control_return(6); } /* Seek to the beginning of the test file. */ status = fx_file_seek(&my_file, 0); /* Check the file seek status. */ if (status != FX_SUCCESS) { printf("ERROR!\n"); test_control_return(7); } /* Clean the buffer. */ memset(read_buffer, 0, read_buffer_size); /* Read the bytes of the test file. */ status = fx_file_read(&my_file, read_buffer, read_buffer_size, &actual); /* Check the file read status. */ if ((status != FX_SUCCESS) || (actual != 28)) { printf("ERROR!\n"); test_control_return(8); } /* Close the test file. */ status = fx_file_close(&my_file); /* Check the file close status. */ if (status != FX_SUCCESS) { printf("ERROR!\n"); test_control_return(9); } /* Create a file called TEST1.TXT in the root directory. */ status = fx_file_create(&ram_disk, "TEST1.TXT"); /* Check the create status. */ if (status != FX_SUCCESS) { printf("ERROR!\n"); test_control_return(10); } /* Open the test file. */ status = fx_file_open(&ram_disk, &my_file, "TEST1.TXT", FX_OPEN_FOR_WRITE); /* Check the file open status. */ if (status != FX_SUCCESS) { printf("ERROR!\n"); test_control_return(11); } /* Write a string to the test file. */ status = fx_file_write(&my_file, (void *) write_buffer, 1000); /* Check the file write status. */ if (status != FX_SUCCESS) { printf("ERROR!\n"); test_control_return(12); } /* Seek to the beginning of the test file. */ status = fx_file_seek(&my_file, 0); /* Check the file seek status. */ if (status != FX_SUCCESS) { printf("ERROR!\n"); test_control_return(13); } /* Clean the buffer. */ memset(read_buffer, 0, read_buffer_size); /* Read the bytes of the test file. */ status = fx_file_read(&my_file, read_buffer, read_buffer_size, &actual); /* Check the file read status. */ if ((status != FX_SUCCESS) || (actual != 1000)) { printf("ERROR!\n"); test_control_return(14); } /* Close the test file. */ status = fx_file_close(&my_file); /* Check the file close status. */ if (status != FX_SUCCESS) { printf("ERROR!\n"); test_control_return(15); } /* Create a file called TEST2.TXT in the root directory. */ status = fx_file_create(&ram_disk, "TEST2.TXT"); /* Check the create status. */ if (status != FX_SUCCESS) { printf("ERROR!\n"); test_control_return(16); } /* Open the test file. */ status = fx_file_open(&ram_disk, &my_file, "TEST2.TXT", FX_OPEN_FOR_WRITE); /* Check the file open status. */ if (status != FX_SUCCESS) { printf("ERROR!\n"); test_control_return(17); } /* Write a string to the test file. */ status = fx_file_write(&my_file, " EXPRESSLOGIC_TEST\n", 19); /* Check the file write status. */ if (status != FX_SUCCESS) { printf("ERROR!\n"); test_control_return(18); } /* Seek to the beginning of the test file. */ status = fx_file_seek(&my_file, 0); /* Check the file seek status. */ if (status != FX_SUCCESS) { printf("ERROR!\n"); test_control_return(19); } /* Clean the buffer. */ memset(read_buffer, 0, read_buffer_size); /* Read the bytes of the test file. */ status = fx_file_read(&my_file, read_buffer, read_buffer_size, &actual); /* Check the file read status. */ if ((status != FX_SUCCESS) || (actual != 19)) { printf("ERROR!\n"); test_control_return(20); } /* Close the test file. */ status = fx_file_close(&my_file); /* Check the file close status. */ if (status != FX_SUCCESS) { printf("ERROR!\n"); test_control_return(21); } /* Create the new thread. */ #ifndef FX_STANDALONE_ENABLE tx_thread_create(&ftest_1, "thread 1", ftest_1_entry, 0, thread_1_buffer, DEMO_STACK_SIZE, 4, 4, TX_NO_TIME_SLICE, TX_AUTO_START); #endif /* directory_write_interrupt */ directory_write_interrupt_1 = FX_FALSE; /* Let the other thread run. */ #ifndef FX_STANDALONE_ENABLE tx_thread_relinquish(); #else pthread_create(&ptid1, NULL, &ftest_1_entry, NULL); usleep(10); pthread_join(ptid1,NULL); #endif /* Delete the thread. */ #ifndef FX_STANDALONE_ENABLE tx_thread_delete(&ftest_1); /* Create the new thread. */ tx_thread_create(&ftest_2, "thread 2", ftest_2_entry, 0, thread_2_buffer, DEMO_STACK_SIZE, 4, 4, TX_NO_TIME_SLICE, TX_AUTO_START); #else pthread_cancel(ptid1); #endif /* directory_write_interrupt */ directory_write_interrupt_2 = FX_FALSE; /* Let the other thread run. */ #ifndef FX_STANDALONE_ENABLE tx_thread_relinquish(); #else pthread_create(&ptid2, NULL, &ftest_2_entry, NULL); usleep(10); pthread_join(ptid2,NULL); #endif /* Recover the log again. */ /* Open the ram_disk. */ status = fx_media_open(&ram_disk, "RAM DISK", _fx_ram_driver, ram_disk_memory_large, cache_buffer, CACHE_SIZE); /* Check the status. */ if (status != FX_SUCCESS) { /* Error, return error code. */ printf("ERROR!\n"); test_control_return(22); } /* Enable the Fault-tolerant feature to recover the media. */ status = fx_fault_tolerant_enable(&ram_disk, fault_tolerant_buffer, FAULT_TOLERANT_SIZE); /* Check status. */ if (status) { printf("ERROR!\n"); test_control_return(23); } /* Open the test file. */ status = fx_file_open(&ram_disk, &my_file, "TEST.TXT", FX_OPEN_FOR_WRITE); /* Check the file open status. */ if (status != FX_SUCCESS) { printf("ERROR!\n"); test_control_return(24); } /* Seek to the beginning of the test file. */ status = fx_file_seek(&my_file, 0); /* Check the file seek status. */ if (status != FX_SUCCESS) { printf("ERROR!\n"); test_control_return(25); } /* Clean the buffer. */ memset(read_buffer, 0, read_buffer_size); /* Read the bytes of the test file. */ status = fx_file_read(&my_file, read_buffer, read_buffer_size, &actual); /* Check the file read status. */ if ((status != FX_SUCCESS) || (actual != 28) || memcmp(read_buffer, " ABCDEFGHIJKLMNOPQRSTUVWXYZ\n", actual)) { printf("ERROR!\n"); test_control_return(26); } /* Close the test file. */ status = fx_file_close(&my_file); /* Check the file close status. */ if (status != FX_SUCCESS) { printf("ERROR!\n"); test_control_return(27); } /* Open the test file. */ status = fx_file_open(&ram_disk, &my_file, "TEST1.TXT", FX_OPEN_FOR_WRITE); /* Check the file open status. */ if (status != FX_SUCCESS) { printf("ERROR!\n"); test_control_return(28); } /* Seek to the beginning of the test file. */ status = fx_file_seek(&my_file, 0); /* Check the file seek status. */ if (status != FX_SUCCESS) { printf("ERROR!\n"); test_control_return(29); } /* Clean the buffer. */ memset(read_buffer, 0, read_buffer_size); /* Read the bytes of the test file. */ status = fx_file_read(&my_file, read_buffer, read_buffer_size, &actual); /* Check the file read status. */ if ((status != FX_SUCCESS) || (actual != 2048 + 1000) || memcmp(read_buffer, write_buffer, actual)) { printf("ERROR!\n"); test_control_return(30); } /* Close the test file. */ status = fx_file_close(&my_file); /* Check the file close status. */ if (status != FX_SUCCESS) { printf("ERROR!\n"); test_control_return(31); } /* Open the test file. */ status = fx_file_open(&ram_disk, &my_file, "TEST2.TXT", FX_OPEN_FOR_WRITE); /* Check the file open status. */ if (status != FX_SUCCESS) { printf("ERROR!\n"); test_control_return(32); } /* Seek to the beginning of the test file. */ status = fx_file_seek(&my_file, 0); /* Check the file seek status. */ if (status != FX_SUCCESS) { printf("ERROR!\n"); test_control_return(33); } /* Clean the buffer. */ memset(read_buffer, 0, read_buffer_size); /* Read the bytes of the test file. */ status = fx_file_read(&my_file, read_buffer, read_buffer_size, &actual); /* Check the file read status. */ if ((status != FX_SUCCESS) || (actual != 19) || memcmp(read_buffer, " EXPRESSLOGIC_TEST\n", actual)) { printf("ERROR!\n"); test_control_return(34); } /* Close the test file. */ status = fx_file_close(&my_file); /* Check the file close status. */ if (status != FX_SUCCESS) { printf("ERROR!\n"); test_control_return(35); } /* Close the media. */ status = fx_media_close(&ram_disk); /* Determine if the test was successful. */ if ((status != FX_SUCCESS) || (directory_write_interrupt_1 != FX_TRUE) || (directory_write_interrupt_2 != FX_TRUE) || (error_couter)) { printf("ERROR!\n"); test_control_return(36); } /* Delete the thread. */ #ifndef FX_STANDALONE_ENABLE tx_thread_delete(&ftest_2); #else pthread_cancel(ptid2); #endif } /* Output successful. */ printf("SUCCESS!\n"); test_control_return(0); } /* Define the test threads. */ #ifndef FX_STANDALONE_ENABLE static void ftest_1_entry(ULONG thread_input) #else void * ftest_1_entry(void * thread_input) #endif { #ifdef FX_STANDALONE_ENABLE UINT oldtype; pthread_setcancelstate(PTHREAD_CANCEL_ENABLE, NULL); pthread_setcanceltype(PTHREAD_CANCEL_ASYNCHRONOUS, &oldtype); #endif UINT status; FX_PARAMETER_NOT_USED(thread_input); /* Set the callback function to simulate poweoff operation when write directory. */ driver_write_callback = my_driver_write_1; /* Open the test file. */ status = fx_file_open(&ram_disk, &my_file, "TEST1.TXT", FX_OPEN_FOR_WRITE); /* Check the file open status. */ if (status != FX_SUCCESS) { error_couter ++; #ifndef FX_STANDALONE_ENABLE return; #else return NULL; #endif } /* Write 2048 bytes to the file, then update the directory. (bytes should be greate than one cluster). */ fx_file_write(&my_file, (void *) (write_buffer + 1000), 2048); } #ifndef FX_STANDALONE_ENABLE static void ftest_2_entry(ULONG thread_input) #else static void * ftest_2_entry(void * thread_input) #endif { #ifdef FX_STANDALONE_ENABLE UINT oldtype; pthread_setcancelstate(PTHREAD_CANCEL_ENABLE, NULL); pthread_setcanceltype(PTHREAD_CANCEL_ASYNCHRONOUS, &oldtype); #endif UINT status; FX_PARAMETER_NOT_USED(thread_input); /* Open the ram_disk. */ status = fx_media_open(&ram_disk, "RAM DISK", _fx_ram_driver, ram_disk_memory_large, cache_buffer, CACHE_SIZE); /* Check the status. */ if (status != FX_SUCCESS) { error_couter ++; #ifndef FX_STANDALONE_ENABLE return; #else return NULL; #endif } /* Set the callback function to simulate poweoff operation when recover the directory. */ driver_write_callback = my_driver_write_2; /* Enable the Fault-tolerant feature to recover the media. */ fx_fault_tolerant_enable(&ram_disk, fault_tolerant_buffer, FAULT_TOLERANT_SIZE); } static UINT my_driver_write_1(FX_MEDIA *media_ptr, UINT sector_type, UCHAR *block_ptr, UINT *operation_ptr) { FX_PARAMETER_NOT_USED(block_ptr); /* Interrupt the directory write operation. */ if ((sector_type == FX_DIRECTORY_SECTOR) && (_filex_fault_tolerant_log_check(media_ptr) & FX_FAULT_TOLERANT_LOG_REDO_DONE)) { /* Set the write interrupt operation. */ *operation_ptr = FX_OP_WRITE_INTERRUPT; /* Update the flag. */ directory_write_interrupt_1 = FX_TRUE; /* Clean the callback function. */ driver_write_callback = FX_NULL; /* Delete the media protection structure if FX_SINGLE_THREAD is not defined. */ #ifndef FX_SINGLE_THREAD #ifndef FX_DONT_CREATE_MUTEX /* Note that the protection is never released. The mutex delete service will handle all threads waiting access to this media control block. */ tx_mutex_delete(&(media_ptr -> fx_media_protect)); #endif #endif /* Clean the media data. */ _fx_system_media_opened_ptr = FX_NULL; _fx_system_media_opened_count = 0; /* Clean the media. */ memset(media_ptr, 0, sizeof(FX_MEDIA)); /* Simulate poweroff. */ /* First terminate the thread to ensure it is ready for deletion. */ #ifndef FX_STANDALONE_ENABLE tx_thread_terminate(&ftest_1); #else pthread_cancel(ptid1); #endif } /* Return. */ return FX_SUCCESS; } static UINT my_driver_write_2(FX_MEDIA *media_ptr, UINT sector_type, UCHAR *block_ptr, UINT *operation_ptr) { FX_PARAMETER_NOT_USED(block_ptr); /* Interrupt the directory write operation. */ if (sector_type == FX_DIRECTORY_SECTOR) { /* Set the write interrupt operation. */ *operation_ptr = FX_OP_WRITE_INTERRUPT; /* Update the flag. */ directory_write_interrupt_2 = FX_TRUE; /* Clean the callback function. */ driver_write_callback = FX_NULL; /* Delete the media protection structure if FX_SINGLE_THREAD is not defined. */ #ifndef FX_SINGLE_THREAD #ifndef FX_DONT_CREATE_MUTEX /* Note that the protection is never released. The mutex delete service will handle all threads waiting access to this media control block. */ tx_mutex_delete(&(media_ptr -> fx_media_protect)); #endif #endif /* Clean the media data. */ _fx_system_media_opened_ptr = FX_NULL; _fx_system_media_opened_count = 0; /* Clean the media. */ memset(media_ptr, 0, sizeof(FX_MEDIA)); /* Simulate poweroff. */ /* First terminate the thread to ensure it is ready for deletion. */ #ifndef FX_STANDALONE_ENABLE tx_thread_terminate(&ftest_2); #else pthread_cancel(ptid2); #endif } /* Return. */ return FX_SUCCESS; } #else #ifdef CTEST void test_application_define(void *first_unused_memory) #else void filex_fault_tolerant_log_recover_directory_test_application_define(void *first_unused_memory) #endif { FX_PARAMETER_NOT_USED(first_unused_memory); /* Print out some test information banners. */ printf("FileX Test: Fault Tolerant Log Recover DIR Test....................N/A\n"); test_control_return(255); } #endif