setup_machine_fdt() -> fixmap_remap_fdt() 부분을 분석하다가

막히는 부분이 생겨 질문드립니다ㅠㅠ

 

fixmap_remap_fdt() 함수를 쭉 따라가다가...


  1   start_kernel          init/main.c
  2   setup_arch          init/main.c
  3   setup_machine_fdt     arch/arm64/kernel/setup.c
  4   fixmap_remap_fdt    arch/arm64/kernel/setup.c
  5   __fixmap_remap_fdt      dt_virt = __fixmap_remap_fdt(dt_phys, &size, PAGE_KERNEL_RO);
  6   create_mapping_noalloc  1221  create_mapping_noalloc(round_down(dt_phys, SWAPPER_BLOCK_SIZE),
  7   __create_pgd_mapping   617  __create_pgd_mapping(init_mm.pgd, phys, virt, size, prot, NULL,
  8   alloc_init_pud    562  alloc_init_pud(pgdp, addr, next, phys, prot, pgtable_alloc,
 

- alloc_init_pud() 부분 까지들어 왔습니다.

 

static void alloc_init_pud(pgd_t *pgdp, unsigned long addr, unsigned long end,
               phys_addr_t phys, pgprot_t prot,
               phys_addr_t (*pgtable_alloc)(void),
               int flags)
{
    unsigned long next;
    pud_t *pudp;
    pgd_t pgd = READ_ONCE(*pgdp);

    if (pgd_none(pgd)) {
        phys_addr_t pud_phys;
        BUG_ON(!pgtable_alloc);
        pud_phys = pgtable_alloc();
        __pgd_populate(pgdp, pud_phys, PUD_TYPE_TABLE);
        pgd = READ_ONCE(*pgdp);
    }
    BUG_ON(pgd_bad(pgd));

    pudp = pud_set_fixmap_offset(pgdp, addr);

 

    do {
        pud_t old_pud = READ_ONCE(*pudp);

        next = pud_addr_end(addr, end);

        /*
         * For 4K granule only, attempt to put down a 1GB block
         */
        if (use_1G_block(addr, next, phys) &&
            (flags & NO_BLOCK_MAPPINGS) == 0) {
            pud_set_huge(pudp, phys, prot);

            /*
             * After the PUD entry has been populated once, we
             * only allow updates to the permission attributes.
             */
            BUG_ON(!pgattr_change_is_safe(pud_val(old_pud),
                              READ_ONCE(pud_val(*pudp))));
        } else {
            alloc_init_cont_pmd(pudp, addr, next, phys, prot,
                        pgtable_alloc, flags);

            BUG_ON(pud_val(old_pud) != 0 &&
                   pud_val(old_pud) != READ_ONCE(pud_val(*pudp)));
        }
        phys += next - addr;
    } while (pudp++, addr = next, addr != end);

    pud_clear_fixmap();
}

 

 

- pud_set_fixmap_offset(pgdp, addr)을 계속 따라 가다보면..

 

pud_set_fixmap(pud_offset_phys(pgdp, addr)

  set_fixmap_offset(FIX_PUD, addr)

    __set_fixmap_offset(idx, phys, FIXMAP_PAGE_NORMAL)

      

 

#define __set_fixmap_offset(idx, phys, flags)               \
({                                  \
      unsigned long ________addr;                 \
      __set_fixmap(idx, phys, flags);                 \
      ________addr = fix_to_virt(idx) + ((phys) & (PAGE_SIZE - 1));   \
      ________addr;                           \
})

 

노란색 음영의 pudp에는 __set_fixmap_offset()에서 반환되는 값인 _______addr

즉, pudp에는 FIX_PUD 페이지에서 phys의 페이지 오프셋에 해당하는 엔트리 주소가 들어간다고 생각했습니다.

 

 

매핑하는 크기가 2M이기 때문에 1G block은 사용하지 않으므로,

아래 부분에서 do-while 구문 -> if-else 구문에서 else가 실행되어

 

            alloc_init_cont_pmd(pudp, addr, next, phys, prot,
                        pgtable_alloc, flags);

 

alloc_init_cont_pmd() 함수로 들어가게 됩니다.

여기서 페이지를 할당하는 함수포인터 인자인 pgtable_alloc = NULL 이기 때문에

pmd 페이지를 할당하지 못하고 pudp엔트리(FIX_PUD 테이블 엔트리)에 값이 들어가있어야 하는 것으로 생각했습니다.

 

그런데, 제가 이해하고 있기로는 FIX_PUD 테이블은 이전 함수들에서 사용한 적이 없어서

FIX_PUD 테이블 엔트리에는 값이 들어가 있지 않는데... ( = 위에서 구한 pudp 엔트리에 들어가 있는 값이 없음)

pmd 페이지도 할당하지 않는 것으로 이해를 해서 매핑이 이상하게 되는 것 같습니다.

어딘가에서 제가 잘못 생각하고 있는 부분이 있는 것 같은데...

 

 

 

정리를 하면...

 

alloc_init_pud() 함수에서

1) pudp 는 FIX_PUD가 가리키는 페이지의 엔트리 주소이다

2) FIX_PUD 페이지는 이전 함수들에서 사용한 적이 없어서 엔트리에는 값이 들어가 있지 않다.

 

alloc_init_cont_pmd() 함수에서

3) pudp를 읽어 값이 없으면 pgtable_alloc함수로 페이지를 할당하는데,

     pgtable_alloc = NULL이어서 페이지를 할당하지도 않는다..

 

 

혹시 제가 이해하고 있는 것 중에 잘못된 부분이 있으면 지적을 부탁드립니다!

 

 

번호 제목 글쓴이 날짜 조회 수
공지 [공지] 프로그래밍 관련 Q&A 게시판 입니다. woos 2016.04.09 22240
517 alloc과 free함수에 대해서 질문입니다. [1] 정재훈 2011.06.28 93626
516 안드로이드 system 폴더안의 파일을 인식못하는데.. [5] 김영일 2011.04.14 69679
515 x86에서 ZONE_DMA 영역의 사용 [1] 홍문화 2011.05.02 58233
514 커널에서 page alloc에 관해서 [1] 미다리로23 2017.02.05 34561
513 디바이스 드라이버 모듈에서 작성한 함수를 커널에서 사용하는 방법 [1] 개짖는소리 2017.02.06 27034
512 Linux kernel의 CFS(Completely Fair Scheduling) 스케줄러에서 time slice 구하는 방법? [10] 김재호 2010.04.30 25221
511 "만들면서 배우는 OS..." 4장 kernel.asm 중 idtr 설정 질문드립니다 쿨쿨 2013.06.18 24943
510 리눅스 스케줄러 관련 [9] 홍문화 2011.02.23 22946
509 NFS NAS SAN 이란 무엇인가? file 박은병 2010.04.13 21880
508 s3c2440기반의 리눅스 커널 분석 중 MM관련 질문입니다. 유태경 2007.09.02 21523
507 /dev 폴더에 있는 node파일 중에서 어떤 것을 open해야 플래쉬를 사용할 수 있을까요? [1] 박종주 2012.02.17 19849
506 sparsemem에 대한 질문이 있습니다. [5] 안정모 2010.03.20 19717
505 16기 리눅스 커널 스터디 C조 18주차 질문입니다. [3] cien 2019.10.12 18822
504 tty, tty0, console, vc 는 각각 뭐하는 디바이스인지? [3] 이창만 2008.12.04 18551
503 ARM A8 이상의 프로세서를 공부할 때 스터디 방향 [4] 유강희 2011.06.15 18511
502 임베디드 OS 만들기 내용과 관련하여 문의드립니다. [5] 송인재 2010.03.25 18321
501 far call 이란 무엇인가요? [4] 박한범 2010.04.13 17840
500 파일시스템에 관한 질문(Htree [6] 사앙조 2017.05.22 17066
499 SMP관련 질문입니다. [2] 컴퓨터 2017.04.13 16583
XE Login