色哟哟视频在线观看-色哟哟视频在线-色哟哟欧美15最新在线-色哟哟免费在线观看-国产l精品国产亚洲区在线观看-国产l精品国产亚洲区久久

0
  • 聊天消息
  • 系統消息
  • 評論與回復
登錄后你可以
  • 下載海量資料
  • 學習在線課程
  • 觀看技術視頻
  • 寫文章/發帖/加入社區
會員中心
創作中心

完善資料讓更多小伙伴認識你,還能領取20積分哦,立即完善>

3天內不再提示

riscv在rt-smart中的板級初始化

RTThread物聯網操作系統 ? 來源:輝輝與海 ? 2023-02-09 17:44 ? 次閱讀

本文章的代碼來自于rt-smart中針對qemu-virt-riscv的bsp

倉庫地址 https://gitee.com/rtthread/rt-thread/tree/rt-smart/

commit ID:d28249c08a152bcf0e1a076cf5b4b082c0a84add

qemu-virt-riscv介紹

簡介

Virt板不對應于任何真實硬件的平臺;它是為虛擬機設計的。如果你只是想運行Linux等客戶機,而不關心重現真實世界硬件的特殊性和局限性,那么它是推薦的板卡類型。

內存空間布局(包括外設地址)

staticconstMemMapEntryvirt_memmap[]={
[VIRT_DEBUG]={0x0,0x100},
[VIRT_MROM]={0x1000,0xf000},
[VIRT_TEST]={0x100000,0x1000},
[VIRT_RTC]={0x101000,0x1000},
[VIRT_CLINT]={0x2000000,0x10000},
[VIRT_ACLINT_SSWI]={0x2F00000,0x4000},
[VIRT_PCIE_PIO]={0x3000000,0x10000},
[VIRT_PLIC]={0xc000000,VIRT_PLIC_SIZE(VIRT_CPUS_MAX*2)},
[VIRT_APLIC_M]={0xc000000,APLIC_SIZE(VIRT_CPUS_MAX)},
[VIRT_APLIC_S]={0xd000000,APLIC_SIZE(VIRT_CPUS_MAX)},
[VIRT_UART0]={0x10000000,0x100},/*串口設備*/
[VIRT_VIRTIO]={0x10001000,0x1000},
[VIRT_FW_CFG]={0x10100000,0x18},
[VIRT_FLASH]={0x20000000,0x4000000},
[VIRT_IMSIC_M]={0x24000000,VIRT_IMSIC_MAX_SIZE},
[VIRT_IMSIC_S]={0x28000000,VIRT_IMSIC_MAX_SIZE},
[VIRT_PCIE_ECAM]={0x30000000,0x10000000},
[VIRT_PCIE_MMIO]={0x40000000,0x40000000},
[VIRT_DRAM]={0x80000000,0x0},/*DDR空間*/
};

rt-smart針對virt board的ddr空間規劃

參考鏈接腳本

bspqemu-virt64-riscvlink.lds

以及board.h中的相關定義

bspqemu-virt64-riscvdriveroard.h

得到ddr的空間規劃如下

內容 地址空間
代碼段數據段棧空間以及bss段 0x80200000 ~ __bss_end
堆空間 __bss_end ~ __bss_end + 100M
頁分配空間 __bss_end + 100M ~ __bss_end + 200M

rt-smart針對virt board的初始化

整體初始化

rt_hw_board_init定義了與qemu-virt-riscv相關的板級初始化的全部內容,包括內存系統,plic中斷子系統,定時器系統以及串口設備等。它由rtthread_startup調用,完整的調用路徑如下。

(libcpu isc-vvirt64startup_gcc.S)_start->primary_cpu_entry->entry->rtthread_startup->rt_hw_board_init

源碼如下

voidrt_hw_board_init(void)
{
#ifdefRT_USING_USERSPACE
rt_page_init(init_page_region);
/*initmmu_infostructure*/
rt_hw_mmu_map_init(&mmu_info,(void*)(USER_VADDR_START-IOREMAP_SIZE),IOREMAP_SIZE,(rt_size_t*)MMUTable,0);
//thisAPIisreservedcurrentlysincePLICetchadnotbeenportingcompletelytoMMUversion
rt_hw_mmu_kernel_map_init(&mmu_info,0x00000000UL,0x80000000);
/*setupregion,andenableMMU*/
rt_hw_mmu_setup(&mmu_info,platform_mem_desc,NUM_MEM_DESC);

#endif

#ifdefRT_USING_HEAP
/*initializememorysystem*/
rt_system_heap_init(RT_HW_HEAP_BEGIN,RT_HW_HEAP_END);
#endif

plic_init();

rt_hw_interrupt_init();

rt_hw_uart_init();

#ifdefRT_USING_CONSOLE
/*setconsoledevice*/
rt_console_set_device(RT_CONSOLE_DEVICE_NAME);
#endif/*RT_USING_CONSOLE*/

rt_hw_tick_init();

#ifdefRT_USING_COMPONENTS_INIT
rt_components_board_init();
#endif

#ifdefRT_USING_HEAP
rt_kprintf("heap:[0x%08x-0x%08x]
",(rt_ubase_t)RT_HW_HEAP_BEGIN,(rt_ubase_t)RT_HW_HEAP_END);
#endif/*RT_USING_HEAP*/
}

rt_page_init

rt-smart中使用了buddy算法管理了一部分內存區域,系統使用page_alloc來向buddy管理的內存區域申請內存資源,像linux一樣每個page是4k的大小。rt-smart采用buddy算法將系統中部分可用的物理內存頁面按照每1個頁面、2個頁面、4個頁面等等劃分為了不同的單元。詳情可參考這篇文章https://club.rt-thread.org/ask/article/3e3a9a0b6d3e2105.html

voidrt_page_init(rt_region_treg)
{
inti;

LOG_D("split0x%08x0x%08x
",reg.start,reg.end);

reg.start+=ARCH_PAGE_MASK;
reg.start&=~ARCH_PAGE_MASK;

reg.end&=~ARCH_PAGE_MASK;

{
intnr=ARCH_PAGE_SIZE/sizeof(structpage);
inttotal=(reg.end-reg.start)>>ARCH_PAGE_SHIFT;
intmnr=(total+nr)/(nr+1);

LOG_D("nr=0x%08x
",nr);
LOG_D("total=0x%08x
",total);
LOG_D("mnr=0x%08x
",mnr);

RT_ASSERT(mnr>ARCH_PAGE_SHIFT;
}

這里rt-smart直接將一部分頁表空間分配給struct page去使用,有可能會造成頁面的浪費。例如當total=7,nr=5時,mnr=2,也就是倆個頁表用于存儲page,五個頁表是真正可以被alloc_page申請的。但實際上五個頁表只需要一個頁表的空間就可以存放page結構體了,相當于浪費了一個頁表。

rt_hw_mmu_map_init

#defineUSER_VADDR_START0x100000000UL
#defineIOREMAP_SIZE(1ul<vtable=vtable;
mmu_info->vstart=va_s;
mmu_info->vend=va_e;
mmu_info->pv_off=pv_off;

return0;
}

mmu_info是一個全局變量,在調用rt_hw_mmu_map_init后,(USER_VADDR_START - IOREMAP_SIZE) ~ USER_VADDR_START 這片虛擬地址空間將來專門提供給ioremap來使用。也就是ioremap返回的虛擬地址區間就是0xc0000000 ~ 0xFFFFFFFF

rt_hw_mmu_kernel_map_init

voidrt_hw_mmu_kernel_map_init(rt_mmu_info*mmu_info,rt_size_tvaddr_start,rt_size_tsize)
{
rt_size_tpaddr_start=__UMASKVALUE(VPN_TO_PPN(vaddr_start,mmu_info->pv_off),PAGE_OFFSET_MASK);
rt_size_tva_s=GET_L1(vaddr_start);
rt_size_tva_e=GET_L1(vaddr_start+size-1);
rt_size_ti;

for(i=va_s;i<=?va_e;?i++)
????{
????????mmu_info->vtable[i]=COMBINEPTE(paddr_start,PAGE_ATTR_RWX|PTE_G|PTE_V);
paddr_start+=L1_PAGE_SIZE;
}

rt_hw_cpu_tlb_invalidate();
}

這里將0x0 ~ 0x80000000的物理地址空間做了offset為0的一比一映射,且只使用了一級頁表。之后0x80000000之下的地址CPU都可以直接訪問了。從頁表的屬性配置上看,這片區域是nocache的。

rt_hw_mmu_setup

#defineKERNEL_VADDR_START0x80000000
#definePV_OFFSET0

structmem_descplatform_mem_desc[]={
{KERNEL_VADDR_START,KERNEL_VADDR_START+0x10000000-1,KERNEL_VADDR_START+PV_OFFSET,NORMAL_MEM},
};

voidrt_hw_mmu_setup(rt_mmu_info*mmu_info,structmem_desc*mdesc,intdesc_nr)
{
void*err;
for(size_ti=0;iattr)
{
caseNORMAL_MEM:
attr=MMU_MAP_K_RWCB;
break;
caseNORMAL_NOCACHE_MEM:
attr=MMU_MAP_K_RWCB;
break;
caseDEVICE_MEM:
attr=MMU_MAP_K_DEVICE;
break;
default:
attr=MMU_MAP_K_DEVICE;
}
rt_kprintf("vaddrstart:%lxpaddr_start:%lx
",mdesc->vaddr_start,mdesc->paddr_start);
err=_rt_hw_mmu_map(mmu_info,(void*)mdesc->vaddr_start,(void*)mdesc->paddr_start,
mdesc->vaddr_end-mdesc->vaddr_start+1,attr);
mdesc++;
}
rt_hw_mmu_switch((void*)MMUTable);
}

這里首先將0x80000000 ~ 0x90000000這片區域做了offset為0的線性映射,映射使用的是三級頁表一頁一頁映射的,相當于page的區域也被映射好了。之后調用rt_hw_mmu_switch配置SATP配置MMU的地址翻譯模式為SV39。STAP的mode被配置后,MMU就相當于開啟了。

將rtconfig.h中的PV_OFFSET改為非0值后系統無法啟動,對比bsp/qemu-vexpress-a9中board.c里關于頁表的配置這塊兒應該還是有問題的。

rt_hw_tick_init

intrt_hw_tick_init(void)
{
/*Readcoreid*/
//unsignedlongcore_id=current_coreid();
unsignedlonginterval=1000/RT_TICK_PER_SECOND;

/*CleartheSupervisor-TimerbitinSIE*/
clear_csr(sie,SIP_STIP);

/*calculatethetickcycles*/
//tick_cycles=interval*sysctl_clock_get_freq(SYSCTL_CLOCK_CPU)/CLINT_CLOCK_DIV/1000ULL-1;
tick_cycles=40000;
/*Settimer*/
sbi_set_timer(get_ticks()+tick_cycles);

/*EnabletheSupervisor-TimerbitinSIE*/
set_csr(sie,SIP_STIP);

return0;
}

這里使用的是riscv中的mtime。mtime是riscv中定義的一個64位的系統計時器,它被要求工作在常開的時鐘域下。內核中使用以下指令可以讀取mtime的值

staticuint64_tget_ticks()
{
__asm____volatile__(
"rdtime%0"
:"=r"(time_elapsed));
returntime_elapsed;
}

補充知識,在qemu中這個時鐘的獲取來源如下

staticinlineint64_tget_clock_realtime(void)
{
structtimevaltv;

gettimeofday(&tv,NULL);
returntv.tv_sec*1000000000LL+(tv.tv_usec*1000);
}

sbi_set_timer并不是設置timer本身的值,而是設置機器模式計時器比較值寄存器MTIMECMPH, MTIMECMPL的值,當系統計時器的值小于等于 {M/STIMECMPH[31:0],M/STIMECMPL[31:0]}的值時不產生中斷;當系統計時器的值大于 {M/STIMECMPH[31:0],M/STIMECMPL[31:0]} 的值時 CLINT產生對應的計時器中斷。它的配置過程為rt-smart將比較寄存器的配置按規則組織為sbi_call的指令,將指令類型指令參數等放入cpu的a0~a7的寄存器,然后調用ecall指令使cpu陷入M態。

sbi_set_timer->SBI_CALL1(SBI_SET_TIMER, 0, val)->sbi_call

static__inlinestructsbi_ret
sbi_call(uint64_targ7,uint64_targ6,uint64_targ0,uint64_targ1,
uint64_targ2,uint64_targ3,uint64_targ4)
{
structsbi_retret;

registeruintptr_ta0__asm("a0")=(uintptr_t)(arg0);
registeruintptr_ta1__asm("a1")=(uintptr_t)(arg1);
registeruintptr_ta2__asm("a2")=(uintptr_t)(arg2);
registeruintptr_ta3__asm("a3")=(uintptr_t)(arg3);
registeruintptr_ta4__asm("a4")=(uintptr_t)(arg4);
registeruintptr_ta6__asm("a6")=(uintptr_t)(arg6);
registeruintptr_ta7__asm("a7")=(uintptr_t)(arg7);

__asm__volatile(
"ecall"
:"+r"(a0),"+r"(a1)
:"r"(a2),"r"(a3),"r"(a4),"r"(a6),"r"(a7)
:"memory");

ret.error=a0;
ret.value=a1;
return(ret);
}

CPU陷入M態后,opensbi會處理這個ecall產生的異常。獲取內核放到寄存器中參數,把新的值賦值給比較值寄存器,并清除計時器中斷

voidsbi_timer_event_start(u64next_event)
{
if(timer_dev&&timer_dev->timer_event_start)
timer_dev->timer_event_start(next_event);
csr_clear(CSR_MIP,MIP_STIP);
csr_set(CSR_MIE,MIP_MTIP);
}

其他

之后的初始化都是原先rt-thread中的內容了,感興趣的讀者可以自行查閱rt-thread官方的《RT-THREAD 編程指南》手冊來學習。另外需要注意的點在plic_init中,plic的寄存器的基地址沒有使用ioremap就直接使用了,這是因為上面描述的0x0 ~ 0x80000000的物理地址空間被做了offset為0的一比一映射。

rt-smart的ioremap實現

void*rt_ioremap(void*paddr,size_tsize)
{
return_ioremap_type(paddr,size,MM_AREA_TYPE_PHY);
}

void*rt_ioremap_nocache(void*paddr,size_tsize)
{
return_ioremap_type(paddr,size,MM_AREA_TYPE_PHY);
}

void*rt_ioremap_cached(void*paddr,size_tsize)
{
return_ioremap_type(paddr,size,MM_AREA_TYPE_PHY_CACHED);
}

rt-smart中的ioremap實際上只分了倆種映射方式,分別是cache和nocache。在當前的qemu-virt64-riscv里,cache的屬性沒有配置到頁表中,我也沒有查qemu的頁表支不支持配置cache,感興趣的讀者請參考C906的相關代碼libcpu/risc-v/t-head/c906/riscv_mmu.h

/*C-SKYextend*/
#definePTE_SEC(1UL<
staticvoid*_ioremap_type(void*paddr,size_tsize,inttype)
{
void*v_addr=NULL;
size_tattr;

switch(type)
{
caseMM_AREA_TYPE_PHY:
attr=MMU_MAP_K_DEVICE;
break;
caseMM_AREA_TYPE_PHY_CACHED:
attr=MMU_MAP_K_RWCB;
break;
default:
returnv_addr;
}

rt_mm_lock();
v_addr=rt_hw_mmu_map(&mmu_info,0,paddr,size,attr);
if(v_addr)
{
intret=lwp_map_area_insert(&k_map_area,(size_t)v_addr,size,type);
if(ret!=0)
{
_iounmap_range(v_addr,size);
v_addr=NULL;
}
}
rt_mm_unlock();
returnv_addr;
}

__ioremap_type中會記錄頁表要配置的屬性然后調用rt_hw_mmu_map進行映射。之后會將映射得到的虛擬地址插入到k_map_area中。

void*_rt_hw_mmu_map(rt_mmu_info*mmu_info,void*v_addr,void*p_addr,rt_size_tsize,rt_size_tattr)
{
/*代碼省略*/
if(v_addr)
{
/*代碼省略*/
}
else
{
vaddr=find_vaddr(mmu_info,pages);
}

if(vaddr)
{
ret=__rt_hw_mmu_map(mmu_info,(void*)vaddr,p_addr,pages,attr);

if(ret==0)
{
rt_hw_cpu_tlb_invalidate();
return(void*)(vaddr|GET_PF_OFFSET((rt_size_t)p_addr));
}
}
return0;
}

ioremap傳入的虛擬地址是0,所以這里先需要調用find_vaddr得到一個可用的虛擬地址。另一個傳入find_vaddr的參數pages代表要要映射的物理內存區域需要多少個page(4K).

staticsize_tfind_vaddr(rt_mmu_info*mmu_info,intpages)
{
size_tloop_pages;
size_tva;
size_tfind_va=0;
intn=0;
size_ti;

loop_pages=(mmu_info->vend-mmu_info->vstart)?(mmu_info->vend-mmu_info->vstart):1;
loop_pages<<=?(ARCH_INDEX_WIDTH?*?2);
????va?=?mmu_info->vstart;
va<<=?(ARCH_PAGE_SHIFT?+?ARCH_INDEX_WIDTH?*?2);
????for?(i?=?0;?i?=pages){
returnfind_va;
}
}
return0;
}

這里會從mmu_info->vstart的虛擬地址開始找,這個地址就是最前面提到的0xC0000000。從0XC0000000開始一個page一個page的去找,看對應的虛擬地址有沒有被映射。如果沒有,那么將va賦值給find_va。之后會繼續往后查找看能不能找到連續的虛擬內存空間大小可以滿足ioremap需要的大小。如果滿足大小最終就返回找到的虛擬地址。總結這個過程就是尋找一塊連續的沒有被映射的大小滿足的虛擬地址空間。

審核編輯:湯梓紅

聲明:本文內容及配圖由入駐作者撰寫或者入駐合作網站授權轉載。文章觀點僅代表作者本人,不代表電子發燒友網立場。文章及其配圖僅供工程師學習之用,如有內容侵權或者其他違規問題,請聯系本站處理。 舉報投訴
  • Linux
    +關注

    關注

    87

    文章

    11322

    瀏覽量

    209901
  • 代碼
    +關注

    關注

    30

    文章

    4803

    瀏覽量

    68774
  • 虛擬機
    +關注

    關注

    1

    文章

    919

    瀏覽量

    28296
  • 初始化
    +關注

    關注

    0

    文章

    50

    瀏覽量

    11905
  • qemu
    +關注

    關注

    0

    文章

    57

    瀏覽量

    5361

原文標題:riscv在rt-smart中的板級初始化

文章出處:【微信號:RTThread,微信公眾號:RTThread物聯網操作系統】歡迎添加關注!文章轉載請注明出處。

收藏 人收藏

    評論

    相關推薦

    Rt-Smartriscv初始化流程

    Virt是一個不對應于任何真實硬件的平臺;它是為虛擬機設計的。如果你只是想運行Linux等客戶機,而不關心重現真實世界硬件的特殊性和局限性,那么它是推薦的板卡類型。
    的頭像 發表于 10-12 14:15 ?1727次閱讀
    <b class='flag-5'>Rt-Smart</b><b class='flag-5'>在</b><b class='flag-5'>riscv</b><b class='flag-5'>中</b>的<b class='flag-5'>初始化</b>流程

    RT-Smart的資料合集

    1、RT-Smart的啟動過程熟悉 RT-Smart 架構的過程,研究其啟動過程的是必不可少的,那么系統正常運行之前,需要做哪些準備工
    發表于 03-22 15:06

    D1哪吒開發rt-smart內核固件的燒寫與運行步驟

    前BSP需要簡單的修改才能保證編譯通過,這里注意講解 rt-smart 內核固件的燒寫與運行搭建d1-allwinner-nezha 的 rt-smart 最小系統后,ubuntu RISC
    發表于 06-17 11:06

    RT-Smartriscv64上的系統初始化和異常處理的代碼注釋

    RT-Smart riscv64匯編注釋以rt-smart全志D1上的代碼為例,主要注釋了rt-smart
    發表于 02-10 16:43

    rt-smartriscv64上的系統初始化和異常處理的代碼注釋

    rt-smart全志D1上的代碼為例,主要注釋了rt-smartriscv64上的系統初始化
    發表于 02-15 11:04

    Rt-Smartriscv初始化流程

    的相關定義bsp\\qemu-virt64-riscv\\driver\\board.h得到ddr的空間規劃如下rt-smart針對virt board的初始化整體初始化
    發表于 02-16 14:09

    樹莓派上rt-smart的應用編程入門

    文章,一些介紹及樹莓派上rt-smart的應用編程入門(更多的從應用程序角度入手)。后續還包括rt-smart上的不同應用程序介紹: wget curl移植 busybox移植 sdl圖形類
    的頭像 發表于 05-13 14:10 ?3201次閱讀
    樹莓派上<b class='flag-5'>rt-smart</b>的應用編程入門

    rt-smart移植分析:從樹莓派3b入手

    移植rt-smart到最新的板子上具體需要注意哪些細節,哪些才是移植rt-smart的關鍵點?本文從樹莓派3b上移植rt-smart的角度,從頭分析rt-sm...
    發表于 01-25 18:48 ?0次下載
    <b class='flag-5'>rt-smart</b>移植分析:從樹莓派3b入手

    RT-Thread自動初始化機制

    ??分析之前首先查閱 RT-Thread 的官方文檔 [RT-Thread 自動初始化機制](https://www.rt-thread.
    的頭像 發表于 06-17 08:52 ?2695次閱讀
    <b class='flag-5'>RT</b>-Thread自動<b class='flag-5'>初始化</b>機制

    RT-Smart初始化相關功能及物理頁分配算法伙伴系統的實現

    想要對 RT-Smart 的物理頁內存管理功能有所了解,需要熟悉相關代碼。
    的頭像 發表于 10-19 10:05 ?1403次閱讀

    優雅的D1S上運行RT-Smart

    前言 最近在學習 RT-Smart ,正巧有全志開發者論壇看到這么一篇帖子【驚】麻雀上運行國產rt-smart系統,看到很多人都在關注 D1S
    的頭像 發表于 11-16 20:15 ?2923次閱讀

    絲滑的RT-Smart用戶態運行LVGL

    /rt-thread.git 更詳細環境配置請移步到— RT-Thread-優雅のD1S上運行RT-Smart 「Rb君」,公眾號:RTThread物聯網操作系統優雅的
    的頭像 發表于 11-22 20:20 ?1282次閱讀

    RT-Smart riscv64匯編注釋

    rt-smart全志D1上的代碼為例,主要注釋了rt-smartriscv64上的系統初始化
    的頭像 發表于 02-08 21:40 ?1178次閱讀

    riscvrt-smart初始化

    本文章的代碼來自于rt-smart針對qemu-virt-riscv的bsp 倉庫地址 https://gitee.com/rtthread/rt-thread/tree/
    的頭像 發表于 02-09 17:45 ?994次閱讀

    RT-Smart riscv64匯編注釋

    rt-smart全志D1上的代碼為例,主要注釋了rt-smartriscv64上的系統初始化
    的頭像 發表于 10-12 17:26 ?631次閱讀
    <b class='flag-5'>RT-Smart</b> <b class='flag-5'>riscv</b>64匯編注釋
    主站蜘蛛池模板: 两个洞一起插哦!好刺激| 久久青青草视频在线观| 麻豆高清区在线| 一本道高清码| 果冻传媒在线观看进入窗口 | 人妻插B视频一区二区三区| 99午夜视频| 欧美成人中文字幕在线看| 99久久国产综合精品成人影院| 男女床上黄色| www.亚洲天堂| 日本无码免费久久久精品| 菲律宾毛片| 小草高清视频免费直播| 国语自产视频在线不卡| 亚洲伊人色综合久久天天伊人| 久久久国产精品免费A片蜜臀| 2021精品高清卡1卡2卡3麻豆| 欧美fxxx| 俄罗斯女人与马Z00Z视频| 亚洲 日韩 在线 国产 精品| 经典WC女厕所里TV| 40分钟超爽大片黄| 千禧金瓶梅快播| 男人的天堂MV在线视频免费观看| 777EY_卡通动漫_1页| 青青青青青青草| 国产欧美国日产在线播放| 亚洲欧美自拍明星换脸| 久久www成人看片| 97在线看视频福利免费| 色色色999| 九九久久国产精品大片| china男士同性视频tv| 武汉美女洗澡| 可以看的黄页的网站| 大香网伊人久久综合观看| 亚洲日韩精品AV中文字幕| 欧美另类jizzhd| 国产亚洲精品久久久久久一区二区| 依人在线观看|