代码拉取完成,页面将自动刷新
/*
* Contributors: Youngjae Kim (youkim@cse.psu.edu)
* Aayush Gupta (axg354@cse.psu.edu)
*
* In case if you have any doubts or questions, kindly write to: youkim@cse.psu.edu
*
* This source file implements the DFTL FTL scheme.
* The detail algorithm for the DFTL can be obtainable from
* "DFTL: A Flash Translation Layer Employing Demand-based * Selective Caching of Page-level Address Mapping", ASPLOS, 2009".
*
*/
#include <stdlib.h>
#include <string.h>
#include "flash.h"
#include "dftl.h"
#include "global.h"
_u32 opm_gc_cost_benefit();
struct omap_dir *mapdir;
blk_t extra_blk_num;
_u32 free_blk_no[2];
_u16 free_page_no[2];
extern int merge_switch_num;
extern int merge_partial_num;
extern int merge_full_num;
extern int page_num_for_2nd_map_table;
int stat_gc_called_num;
double total_gc_overhead_time;
int map_pg_read=0;
_u32 opm_gc_cost_benefit()
{
int max_cb = 0;
int blk_cb;
_u32 max_blk = -1, i;
for (i = 0; i < nand_blk_num; i++) {
if(i == free_blk_no[0] || i == free_blk_no[1]){
continue;
}
blk_cb = nand_blk[i].ipc;
if (blk_cb > max_cb) {
max_cb = blk_cb;
max_blk = i;
}
}
ASSERT(max_blk != -1);
ASSERT(nand_blk[max_blk].ipc > 0);
return max_blk;
}
size_t opm_read(sect_t lsn, sect_t size, int mapdir_flag)
{
int i;
int lpn = lsn/SECT_NUM_PER_PAGE; // logical page number
int size_page = size/SECT_NUM_PER_PAGE; // size in page
int sect_num;
sect_t s_lsn; // starting logical sector number
sect_t s_psn; // starting physical sector number
sect_t lsns[SECT_NUM_PER_PAGE];
ASSERT(lpn < opagemap_num);
ASSERT(lpn + size_page <= opagemap_num);
memset (lsns, 0xFF, sizeof (lsns));
sect_num = (size < SECT_NUM_PER_PAGE) ? size : SECT_NUM_PER_PAGE;
if(mapdir_flag == 2){
s_psn = mapdir[lpn].ppn * SECT_NUM_PER_PAGE;
}
else s_psn = opagemap[lpn].ppn * SECT_NUM_PER_PAGE;
s_lsn = lpn * SECT_NUM_PER_PAGE;
for (i = 0; i < SECT_NUM_PER_PAGE; i++) {
lsns[i] = s_lsn + i;
}
if(mapdir_flag == 2){
map_pg_read++;
}
size = nand_page_read(s_psn, lsns, 0);
ASSERT(size == SECT_NUM_PER_PAGE);
return sect_num;
}
int opm_gc_get_free_blk(int small, int mapdir_flag)
{
if (free_page_no[small] >= SECT_NUM_PER_BLK) {
free_blk_no[small] = nand_get_free_blk(1);
free_page_no[small] = 0;
return -1;
}
return 0;
}
int opm_gc_run(int small, int mapdir_flag)
{
blk_t victim_blk_no;
int merge_count;
int i,z, j,m,q, benefit = 0;
int k,old_flag,temp_arr[PAGE_NUM_PER_BLK],temp_arr1[PAGE_NUM_PER_BLK],map_arr[PAGE_NUM_PER_BLK];
int valid_flag,pos;
_u32 copy_lsn[SECT_NUM_PER_PAGE], copy[SECT_NUM_PER_PAGE];
_u16 valid_sect_num, l, s;
victim_blk_no = opm_gc_cost_benefit();
memset(copy_lsn, 0xFF, sizeof (copy_lsn));
s = k = OFF_F_SECT(free_page_no[small]);
if(!((s == 0) && (k == 0))){
printf("s && k should be 0\n");
exit(0);
}
small = -1;
//根本就没有用到输入的参量small
for( q = 0; q < PAGE_NUM_PER_BLK; q++){
if(nand_blk[victim_blk_no].page_status[q] == 1){ //map block
for( q = 0; q < 64; q++) {
if(nand_blk[victim_blk_no].page_status[q] == 0 ){
printf("something corrupted1=%d",victim_blk_no);
}
}
small = 0;
break;
}
else if(nand_blk[victim_blk_no].page_status[q] == 0){ //data block
for( q = 0; q < 64; q++) {
if(nand_blk[victim_blk_no].page_status[q] == 1 ){
printf("something corrupted2 %d \n",victim_blk_no);
}
}
small = 1;
break;
}
}
ASSERT ( small == 0 || small == 1);
pos = 0;
merge_count = 0;
for (i = 0; i < PAGE_NUM_PER_BLK; i++)
{
valid_flag = Flash_NandOOBRead( SECTOR(victim_blk_no, i * SECT_NUM_PER_PAGE));
if(valid_flag == 1)
{
valid_sect_num = nand_page_read( SECTOR(victim_blk_no, i * SECT_NUM_PER_PAGE), copy, 1);
merge_count++;
ASSERT(valid_sect_num == 4);
k=0;
for (j = 0; j < valid_sect_num; j++) {
copy_lsn[k] = copy[j];
k++;
}
benefit += opm_gc_get_free_blk(small, mapdir_flag);
if(nand_blk[victim_blk_no].page_status[i] == 1)
{
mapdir[(copy_lsn[s]/SECT_NUM_PER_PAGE)].ppn = BLK_PAGE_NO_SECT(SECTOR(free_blk_no[small], free_page_no[small]));
opagemap[copy_lsn[s]/SECT_NUM_PER_PAGE].ppn = BLK_PAGE_NO_SECT(SECTOR(free_blk_no[small], free_page_no[small]));
nand_page_write(SECTOR(free_blk_no[small],free_page_no[small]) & (~OFF_MASK_SECT), copy_lsn, 1, 2);
free_page_no[small] += SECT_NUM_PER_PAGE;
}
else{
//如果有效页存的是数据页,就比较复杂了,需要考虑同步更新对应CMT中的映射关系
opagemap[BLK_PAGE_NO_SECT(copy_lsn[s])].ppn = BLK_PAGE_NO_SECT(SECTOR(free_blk_no[small], free_page_no[small]));
nand_page_write(SECTOR(free_blk_no[small],free_page_no[small]) & (~OFF_MASK_SECT), copy_lsn, 1, 1);
free_page_no[small] += SECT_NUM_PER_PAGE;
//如果修改的数据页的映射关系存在CMT中,则延迟更改对应的翻译页
if((opagemap[BLK_PAGE_NO_SECT(copy_lsn[s])].map_status == MAP_REAL) || (opagemap[BLK_PAGE_NO_SECT(copy_lsn[s])].map_status == MAP_GHOST)) {
delay_flash_update++;
}
else {
//反之则保留lsn到map_arr中 ,map_arr[PAGE_NUM_PER_BLK]
map_arr[pos] = copy_lsn[s];
pos++;
}
}
}
}
for(i=0;i < PAGE_NUM_PER_BLK;i++) {
temp_arr[i]=-1;
}
k=0;
for(i =0 ; i < pos; i++) {
old_flag = 0;
for( j = 0 ; j < k; j++) {
if(temp_arr[j] == mapdir[((map_arr[i]/SECT_NUM_PER_PAGE)/MAP_ENTRIES_PER_PAGE)].ppn) {
if(temp_arr[j] == -1){
printf("something wrong");
ASSERT(0);
}
old_flag = 1;
break;
}
}
if( old_flag == 0 ) {
temp_arr[k] = mapdir[((map_arr[i]/SECT_NUM_PER_PAGE)/MAP_ENTRIES_PER_PAGE)].ppn;
temp_arr1[k] = map_arr[i];
k++;
}
else
save_count++;
}
for ( i=0; i < k; i++) {
if (free_page_no[0] >= SECT_NUM_PER_BLK) {
if((free_blk_no[0] = nand_get_free_blk(1)) == -1){
printf("we are in big trouble shudnt happen");
}
free_page_no[0] = 0;
}
nand_page_read(temp_arr[i]*SECT_NUM_PER_PAGE,copy,1);
//将整个页的扇区进行无效置位
for(m = 0; m<SECT_NUM_PER_PAGE; m++){
nand_invalidate(mapdir[((temp_arr1[i]/SECT_NUM_PER_PAGE)/MAP_ENTRIES_PER_PAGE)].ppn*SECT_NUM_PER_PAGE+m, copy[m]);
}
nand_stat(OOB_WRITE);
mapdir[((temp_arr1[i]/SECT_NUM_PER_PAGE)/MAP_ENTRIES_PER_PAGE)].ppn = BLK_PAGE_NO_SECT(SECTOR(free_blk_no[0], free_page_no[0]));
opagemap[((temp_arr1[i]/SECT_NUM_PER_PAGE)/MAP_ENTRIES_PER_PAGE)].ppn = BLK_PAGE_NO_SECT(SECTOR(free_blk_no[0], free_page_no[0]));
nand_page_write(SECTOR(free_blk_no[0],free_page_no[0]) & (~OFF_MASK_SECT), copy, 1, 2);
free_page_no[0] += SECT_NUM_PER_PAGE;
}
if(merge_count == 0 )
merge_switch_num++;
else if(merge_count > 0 && merge_count < PAGE_NUM_PER_BLK)
merge_partial_num++;
else if(merge_count == PAGE_NUM_PER_BLK)
merge_full_num++;
else if(merge_count > PAGE_NUM_PER_BLK){
printf("merge_count =%d PAGE_NUM_PER_BLK=%d",merge_count,PAGE_NUM_PER_BLK);
ASSERT(0);
}
nand_erase(victim_blk_no);
return (benefit + 1);
}
size_t opm_write(sect_t lsn, sect_t size, int mapdir_flag)
{
int i;
int lpn = lsn/SECT_NUM_PER_PAGE; // logical page number
int size_page = size/SECT_NUM_PER_PAGE; // size in page
int ppn;
int small;
sect_t lsns[SECT_NUM_PER_PAGE];
int sect_num = SECT_NUM_PER_PAGE;
sect_t s_lsn; // starting logical sector number
sect_t s_psn; // starting physical sector number
sect_t s_psn1;
ASSERT(lpn < opagemap_num);
ASSERT(lpn + size_page <= opagemap_num);
s_lsn = lpn * SECT_NUM_PER_PAGE;
if(mapdir_flag == 2) //map page
small = 0;
else if ( mapdir_flag == 1) //data page
small = 1;
else{
printf("something corrupted");
exit(0);
}
if (free_page_no[small] >= SECT_NUM_PER_BLK)
{
if ((free_blk_no[small] = nand_get_free_blk(0)) == -1)
{
int j = 0;
//只有free_blk_num小于3才启用opm_gc_run
while (free_blk_num < 4 ){
j += opm_gc_run(small, mapdir_flag);
}
opm_gc_get_free_blk(small, mapdir_flag);
}
else {
free_page_no[small] = 0;
}
}
memset (lsns, 0xFF, sizeof (lsns));
s_psn = SECTOR(free_blk_no[small], free_page_no[small]);
if(s_psn % 4 != 0){
printf("s_psn: %d\n", s_psn);
}
ppn = s_psn / SECT_NUM_PER_PAGE;
if (opagemap[lpn].free == 0) {
s_psn1 = opagemap[lpn].ppn * SECT_NUM_PER_PAGE;
for(i = 0; i<SECT_NUM_PER_PAGE; i++){
nand_invalidate(s_psn1 + i, s_lsn + i);
}
nand_stat(3);
}
else {
opagemap[lpn].free = 0;
}
for (i = 0; i < SECT_NUM_PER_PAGE; i++)
{
lsns[i] = s_lsn + i;
}
if(mapdir_flag == 2) {
mapdir[lpn].ppn = ppn;
opagemap[lpn].ppn = ppn;
}
else {
opagemap[lpn].ppn = ppn;
}
free_page_no[small] += SECT_NUM_PER_PAGE;
nand_page_write(s_psn, lsns, 0, mapdir_flag);
return sect_num;
}
void opm_end()
{
if (opagemap != NULL) {
free(opagemap);
free(mapdir);
}
opagemap_num = 0;
// if(CFLRU_Head!=NULL){
// FreeList(&CFLRU_Head);
// }
}
void opagemap_reset()
{
cache_hit = 0;
flash_hit = 0;
disk_hit = 0;
evict = 0;
delay_flash_update = 0;
read_count =0;
write_count=0;
map_pg_read=0;
save_count = 0;
}
int opm_init(blk_t blk_num, blk_t extra_num)
{
int i;
int mapdir_num;
opagemap_num = blk_num * PAGE_NUM_PER_BLK;
//create primary mapping table
opagemap = (struct opm_entry *) malloc(sizeof (struct opm_entry) * opagemap_num);
mapdir = (struct omap_dir *)malloc(sizeof(struct omap_dir) * opagemap_num / MAP_ENTRIES_PER_PAGE);
if ((opagemap == NULL) || (mapdir == NULL)) {
return -1;
}
mapdir_num = (opagemap_num / MAP_ENTRIES_PER_PAGE);
if((opagemap_num % MAP_ENTRIES_PER_PAGE) != 0){
printf("opagemap_num % MAP_ENTRIES_PER_PAGE is not zero\n");
mapdir_num++;
}
memset(opagemap, 0xFF, sizeof (struct opm_entry) * opagemap_num);
memset(mapdir, 0xFF, sizeof (struct omap_dir) * mapdir_num);
//youkim: 1st map table
TOTAL_MAP_ENTRIES = opagemap_num;
for(i = 0; i<TOTAL_MAP_ENTRIES; i++){
opagemap[i].cache_status = 0;
opagemap[i].cache_age = 0;
opagemap[i].map_status = 0;
opagemap[i].map_age = 0;
opagemap[i].update = 0;
}
extra_blk_num = extra_num;
free_blk_no[0] = nand_get_free_blk(0);
free_page_no[0] = 0;
free_blk_no[1] = nand_get_free_blk(0);
free_page_no[1] = 0;
//initialize variables
MAP_REAL_NUM_ENTRIES = 0;
MAP_GHOST_NUM_ENTRIES = 0;
CACHE_NUM_ENTRIES = 0;
SYNC_NUM = 0;
cache_hit = 0;
flash_hit = 0;
disk_hit = 0;
evict = 0;
read_cache_hit = 0;
write_cache_hit = 0;
write_count =0;
read_count = 0;
save_count = 0;
//update 2nd mapping table
for(i = 0; i<mapdir_num; i++){
ASSERT(MAP_ENTRIES_PER_PAGE == 512);
opm_write(i*SECT_NUM_PER_PAGE, SECT_NUM_PER_PAGE, 2);
}
/*
for(i = mapdir_num; i<(opagemap_num - mapdir_num - (extra_num * PAGE_NUM_PER_BLK)); i++){
opm_write(i*SECT_NUM_PER_PAGE, SECT_NUM_PER_PAGE, 1);
}
*/
// update dm_table
/*
int j;
for(i = mapdir_num; i<(opagemap_num - mapdir_num - (extra_num * PAGE_NUM_PER_BLK)); i++){
for(j=0; j < SECT_NUM_PER_PAGE;j++)
dm_table[ (i*SECT_NUM_PER_PAGE) + j] = DEV_FLASH;
}
*/
return 0;
}
int opm_invalid(int secno)
{
int lpn = secno/SECT_NUM_PER_PAGE + page_num_for_2nd_map_table;
int s_lsn = lpn * SECT_NUM_PER_PAGE;
int i, s_psn1;
s_psn1 = opagemap[lpn].ppn * SECT_NUM_PER_PAGE;
for(i = 0; i<SECT_NUM_PER_PAGE; i++){
nand_invalidate(s_psn1 + i, s_lsn + i);
}
opagemap[lpn].ppn = -1;
opagemap[lpn].map_status = 0;
opagemap[lpn].map_age = 0;
opagemap[lpn].update = 0;
return SECT_NUM_PER_PAGE;
}
struct ftl_operation opm_operation = {
init: opm_init,
read: opm_read,
write: opm_write,
end: opm_end
};
struct ftl_operation * opm_setup()
{
return &opm_operation;
}
此处可能存在不合适展示的内容,页面不予展示。您可通过相关编辑功能自查并修改。
如您确认内容无涉及 不当用语 / 纯广告导流 / 暴力 / 低俗色情 / 侵权 / 盗版 / 虚假 / 无价值内容或违法国家有关法律法规的内容,可点击提交进行申诉,我们将尽快为您处理。