亚洲欧美第一页_禁久久精品乱码_粉嫩av一区二区三区免费野_久草精品视频

? 歡迎來到蟲蟲下載站! | ?? 資源下載 ?? 資源專輯 ?? 關于我們
? 蟲蟲下載站

?? mtrr.c

?? xen 3.2.2 源碼
?? C
?? 第 1 頁 / 共 2 頁
字號:
/* * mtrr.c: MTRR/PAT virtualization * * Copyright (c) 2007, Intel Corporation. * * This program is free software; you can redistribute it and/or modify it * under the terms and conditions of the GNU General Public License, * version 2, as published by the Free Software Foundation. * * This program is distributed in the hope it will be useful, but WITHOUT * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or * FITNESS FOR A PARTICULAR PURPOSE.  See the GNU General Public License for * more details. * * You should have received a copy of the GNU General Public License along with * this program; if not, write to the Free Software Foundation, Inc., 59 Temple * Place - Suite 330, Boston, MA 02111-1307 USA. */#include <public/hvm/e820.h>#include <xen/types.h>#include <asm/e820.h>#include <asm/paging.h>#include <asm/p2m.h>#include <xen/domain_page.h>#include <asm/mtrr.h>#include <asm/hvm/support.h>#include <asm/hvm/cacheattr.h>/* Xen holds the native MTRR MSRs */extern struct mtrr_state mtrr_state;static uint64_t phys_base_msr_mask;static uint64_t phys_mask_msr_mask;static uint32_t size_or_mask;static uint32_t size_and_mask;static void init_pat_entry_tbl(uint64_t pat);static void init_mtrr_epat_tbl(void);static uint8_t get_mtrr_type(struct mtrr_state *m, paddr_t pa);/* get page attribute fields (PAn) from PAT MSR */#define pat_cr_2_paf(pat_cr,n)  ((((uint64_t)pat_cr) >> ((n)<<3)) & 0xff)/* pat entry to PTE flags (PAT, PCD, PWT bits) */static uint8_t pat_entry_2_pte_flags[8] = {    0,           _PAGE_PWT,    _PAGE_PCD,   _PAGE_PCD | _PAGE_PWT,    _PAGE_PAT,   _PAGE_PAT | _PAGE_PWT,    _PAGE_PAT | _PAGE_PCD, _PAGE_PAT | _PAGE_PCD | _PAGE_PWT };/* effective mm type lookup table, according to MTRR and PAT */static uint8_t mm_type_tbl[MTRR_NUM_TYPES][PAT_TYPE_NUMS] = {/********PAT(UC,WC,RS,RS,WT,WP,WB,UC-)*//* RS means reserved type(2,3), and type is hardcoded here */ /*MTRR(UC):(UC,WC,RS,RS,UC,UC,UC,UC)*/            {0, 1, 2, 2, 0, 0, 0, 0}, /*MTRR(WC):(UC,WC,RS,RS,UC,UC,WC,WC)*/            {0, 1, 2, 2, 0, 0, 1, 1}, /*MTRR(RS):(RS,RS,RS,RS,RS,RS,RS,RS)*/            {2, 2, 2, 2, 2, 2, 2, 2}, /*MTRR(RS):(RS,RS,RS,RS,RS,RS,RS,RS)*/            {2, 2, 2, 2, 2, 2, 2, 2}, /*MTRR(WT):(UC,WC,RS,RS,WT,WP,WT,UC)*/            {0, 1, 2, 2, 4, 5, 4, 0}, /*MTRR(WP):(UC,WC,RS,RS,WT,WP,WP,WC)*/            {0, 1, 2, 2, 4, 5, 5, 1}, /*MTRR(WB):(UC,WC,RS,RS,WT,WP,WB,UC)*/            {0, 1, 2, 2, 4, 5, 6, 0}};/* reverse lookup table, to find a pat type according to MTRR and effective * memory type. This table is dynamically generated */static uint8_t mtrr_epat_tbl[MTRR_NUM_TYPES][MEMORY_NUM_TYPES];/* lookup table for PAT entry of a given PAT value in host pat */static uint8_t pat_entry_tbl[PAT_TYPE_NUMS];static void get_mtrr_range(uint64_t base_msr, uint64_t mask_msr,                           uint64_t *base, uint64_t *end){    uint32_t mask_lo = (uint32_t)mask_msr;    uint32_t mask_hi = (uint32_t)(mask_msr >> 32);    uint32_t base_lo = (uint32_t)base_msr;    uint32_t base_hi = (uint32_t)(base_msr >> 32);    uint32_t size;    if ( (mask_lo & 0x800) == 0 )    {        /* Invalid (i.e. free) range */        *base = 0;        *end = 0;        return;    }    /* Work out the shifted address mask. */    mask_lo = (size_or_mask | (mask_hi << (32 - PAGE_SHIFT)) |               (mask_lo >> PAGE_SHIFT));    /* This works correctly if size is a power of two (a contiguous range). */    size = -mask_lo;    *base = base_hi << (32 - PAGE_SHIFT) | base_lo >> PAGE_SHIFT;    *end = *base + size - 1;}bool_t is_var_mtrr_overlapped(struct mtrr_state *m){    int32_t seg, i;    uint64_t phys_base, phys_mask, phys_base_pre, phys_mask_pre;    uint64_t base_pre, end_pre, base, end;    uint8_t num_var_ranges = (uint8_t)m->mtrr_cap;    for ( i = 0; i < num_var_ranges; i++ )    {        phys_base_pre = ((uint64_t*)m->var_ranges)[i*2];        phys_mask_pre = ((uint64_t*)m->var_ranges)[i*2 + 1];        get_mtrr_range(phys_base_pre, phys_mask_pre,                        &base_pre, &end_pre);        for ( seg = i + 1; seg < num_var_ranges; seg ++ )        {            phys_base = ((uint64_t*)m->var_ranges)[seg*2];            phys_mask = ((uint64_t*)m->var_ranges)[seg*2 + 1];            get_mtrr_range(phys_base, phys_mask,                            &base, &end);            if ( ((base_pre != end_pre) && (base != end))                 || ((base >= base_pre) && (base <= end_pre))                 || ((end >= base_pre) && (end <= end_pre))                 || ((base_pre >= base) && (base_pre <= end))                 || ((end_pre >= base) && (end_pre <= end)) )            {                /* MTRR is overlapped. */                return 1;            }        }    }    return 0;}/* reserved mtrr for guest OS */#define RESERVED_MTRR 2#define MTRRphysBase_MSR(reg) (0x200 + 2 * (reg))#define MTRRphysMask_MSR(reg) (0x200 + 2 * (reg) + 1)bool_t mtrr_var_range_msr_set(struct mtrr_state *m, uint32_t msr,                              uint64_t msr_content);bool_t mtrr_def_type_msr_set(struct mtrr_state *m, uint64_t msr_content);bool_t mtrr_fix_range_msr_set(struct mtrr_state *m, uint32_t row,                              uint64_t msr_content);static void set_var_mtrr(uint32_t reg, struct mtrr_state *m,                         uint32_t base, uint32_t size,                         uint32_t type){    struct mtrr_var_range *vr;    vr = &m->var_ranges[reg];    if ( size == 0 )    {        /* The invalid bit is kept in the mask, so we simply clear the         * relevant mask register to disable a range.         */        mtrr_var_range_msr_set(m, MTRRphysMask_MSR(reg), 0);    }    else    {        vr->base_lo = base << PAGE_SHIFT | type;        vr->base_hi = (base & size_and_mask) >> (32 - PAGE_SHIFT);        vr->mask_lo = -size << PAGE_SHIFT | 0x800;        vr->mask_hi = (-size & size_and_mask) >> (32 - PAGE_SHIFT);        mtrr_var_range_msr_set(m, MTRRphysBase_MSR(reg), *(uint64_t *)vr);        mtrr_var_range_msr_set(m, MTRRphysMask_MSR(reg),                               *((uint64_t *)vr + 1));    }}/* From Intel Vol. III Section 10.11.4, the Range Size and Base Alignment has * some kind of requirement: * 1. The range size must be 2^N byte for N >= 12 (i.e 4KB minimum). * 2. The base address must be 2^N aligned, where the N here is equal to * the N in previous requirement. So a 8K range must be 8K aligned not 4K aligned. */static uint32_t range_to_mtrr(uint32_t reg, struct mtrr_state *m,                              uint32_t range_startk, uint32_t range_sizek,                              uint8_t type){    if ( !range_sizek || (reg >= ((m->mtrr_cap & 0xff) - RESERVED_MTRR)) )    {        gdprintk(XENLOG_WARNING,                "Failed to init var mtrr msr[%d]"                "range_size:%x, total available MSR:%d\n",                reg, range_sizek,                (uint32_t)((m->mtrr_cap & 0xff) - RESERVED_MTRR));        return reg;    }    while ( range_sizek )    {        uint32_t max_align, align, sizek;        max_align = (range_startk == 0) ? 32 : ffs(range_startk);        align = min_t(uint32_t, fls(range_sizek), max_align);        sizek = 1 << (align - 1);        set_var_mtrr(reg++, m, range_startk, sizek, type);        range_startk += sizek;        range_sizek  -= sizek;        if ( reg >= ((m->mtrr_cap & 0xff) - RESERVED_MTRR) )        {            gdprintk(XENLOG_WARNING,                    "Failed to init var mtrr msr[%d],"                    "total available MSR:%d\n",                    reg, (uint32_t)((m->mtrr_cap & 0xff) - RESERVED_MTRR));            break;        }    }    return reg;}static void setup_fixed_mtrrs(struct vcpu *v){    uint64_t content;    int32_t i;    struct mtrr_state *m = &v->arch.hvm_vcpu.mtrr;    /* 1. Map (0~A0000) as WB */    content = 0x0606060606060606ull;    mtrr_fix_range_msr_set(m, 0, content);    mtrr_fix_range_msr_set(m, 1, content);    /* 2. Map VRAM(A0000~C0000) as WC */    content = 0x0101010101010101;    mtrr_fix_range_msr_set(m, 2, content);    /* 3. Map (C0000~100000) as UC */    for ( i = 3; i < 11; i++)        mtrr_fix_range_msr_set(m, i, 0);}static void setup_var_mtrrs(struct vcpu *v){    p2m_type_t p2m;    uint64_t e820_mfn;    int8_t *p = NULL;    uint8_t nr = 0;    int32_t i;    uint32_t reg = 0;    uint64_t size = 0;    uint64_t addr = 0;    struct e820entry *e820_table;    e820_mfn = mfn_x(gfn_to_mfn(v->domain,                    HVM_E820_PAGE >> PAGE_SHIFT, &p2m));    p = (int8_t *)map_domain_page(e820_mfn);    nr = *(uint8_t*)(p + HVM_E820_NR_OFFSET);    e820_table = (struct e820entry*)(p + HVM_E820_OFFSET);    /* search E820 table, set MTRR for RAM */    for ( i = 0; i < nr; i++)    {        if ( (e820_table[i].addr >= 0x100000) &&             (e820_table[i].type == E820_RAM) )        {            if ( e820_table[i].addr == 0x100000 )            {                size = e820_table[i].size + 0x100000 + PAGE_SIZE * 4;                addr = 0;            }            else            {                /* Larger than 4G */                size = e820_table[i].size;                addr = e820_table[i].addr;            }            reg = range_to_mtrr(reg, &v->arch.hvm_vcpu.mtrr,                                addr >> PAGE_SHIFT, size >> PAGE_SHIFT,                                MTRR_TYPE_WRBACK);        }    }}void init_mtrr_in_hyper(struct vcpu *v){    /* TODO:MTRR should be initialized in BIOS or other places.     * workaround to do it in here     */    if ( v->arch.hvm_vcpu.mtrr.is_initialized )        return;    setup_fixed_mtrrs(v);    setup_var_mtrrs(v);    /* enable mtrr */    mtrr_def_type_msr_set(&v->arch.hvm_vcpu.mtrr, 0xc00);    v->arch.hvm_vcpu.mtrr.is_initialized = 1;}static int32_t reset_mtrr(struct mtrr_state *m){    m->var_ranges = xmalloc_array(struct mtrr_var_range, MTRR_VCNT);    if ( m->var_ranges == NULL )        return -ENOMEM;    memset(m->var_ranges, 0, MTRR_VCNT * sizeof(struct mtrr_var_range));    memset(m->fixed_ranges, 0, sizeof(m->fixed_ranges));    m->enabled = 0;    m->def_type = 0;/*mtrr is disabled*/    m->mtrr_cap = (0x5<<8)|MTRR_VCNT;/*wc,fix enabled, and vcnt=8*/    m->overlapped = 0;    return 0;}/* init global variables for MTRR and PAT */void global_init_mtrr_pat(void){    extern uint64_t host_pat;    uint32_t phys_addr;    init_mtrr_epat_tbl();    init_pat_entry_tbl(host_pat);    /* Get max physical address, set some global variable */    if ( cpuid_eax(0x80000000) < 0x80000008 )        phys_addr = 36;    else        phys_addr = cpuid_eax(0x80000008);    phys_base_msr_mask = ~((((uint64_t)1) << phys_addr) - 1) | 0xf00UL;    phys_mask_msr_mask = ~((((uint64_t)1) << phys_addr) - 1) | 0x7ffUL;    size_or_mask = ~((1 << (phys_addr - PAGE_SHIFT)) - 1);    size_and_mask = ~size_or_mask & 0xfff00000;}static void init_pat_entry_tbl(uint64_t pat){    int32_t i, j;    memset(&pat_entry_tbl, INVALID_MEM_TYPE,           PAT_TYPE_NUMS * sizeof(pat_entry_tbl[0]));    for ( i = 0; i < PAT_TYPE_NUMS; i++ )    {        for ( j = 0; j < PAT_TYPE_NUMS; j++ )        {            if ( pat_cr_2_paf(pat, j) == i )            {                pat_entry_tbl[i] = j;                break;            }        }    }}uint8_t pat_type_2_pte_flags(uint8_t pat_type){    int32_t pat_entry = pat_entry_tbl[pat_type];    /* INVALID_MEM_TYPE, means doesn't find the pat_entry in host pat for     * a given pat_type. If host pat covers all the pat types,     * it can't happen.     */    if ( likely(pat_entry != INVALID_MEM_TYPE) )        return pat_entry_2_pte_flags[pat_entry];    return pat_entry_2_pte_flags[pat_entry_tbl[PAT_TYPE_UNCACHABLE]];}int32_t reset_vmsr(struct mtrr_state *m, uint64_t *pat_ptr){    int32_t rc;    rc = reset_mtrr(m);    if ( rc != 0 )        return rc;    *pat_ptr = ((uint64_t)PAT_TYPE_WRBACK) |               /* PAT0: WB */               ((uint64_t)PAT_TYPE_WRTHROUGH << 8) |       /* PAT1: WT */               ((uint64_t)PAT_TYPE_UC_MINUS << 16) |       /* PAT2: UC- */               ((uint64_t)PAT_TYPE_UNCACHABLE << 24) |     /* PAT3: UC */               ((uint64_t)PAT_TYPE_WRBACK << 32) |         /* PAT4: WB */               ((uint64_t)PAT_TYPE_WRTHROUGH << 40) |      /* PAT5: WT */               ((uint64_t)PAT_TYPE_UC_MINUS << 48) |       /* PAT6: UC- */               ((uint64_t)PAT_TYPE_UNCACHABLE << 56);      /* PAT7: UC */    return 0;}/* * Get MTRR memory type for physical address pa. */static uint8_t get_mtrr_type(struct mtrr_state *m, paddr_t pa){   int32_t     addr, seg, index;   uint8_t     overlap_mtrr = 0;   uint8_t     overlap_mtrr_pos = 0;   uint64_t    phys_base;   uint64_t    phys_mask;   uint8_t     num_var_ranges = m->mtrr_cap & 0xff;   if ( unlikely(!(m->enabled & 0x2)) )       return MTRR_TYPE_UNCACHABLE;   if ( (pa < 0x100000) && (m->enabled & 1) )   {       /* Fixed range MTRR takes effective */       addr = (uint32_t) pa;       if ( addr < 0x80000 )       {           seg = (addr >> 16);           return m->fixed_ranges[seg];       }       else if ( addr < 0xc0000 )       {           seg = (addr - 0x80000) >> 14;           index = (seg >> 3) + 1;           seg &= 7;            /* select 0-7 segments */           return m->fixed_ranges[index*8 + seg];       }       else       {           /* 0xC0000 --- 0x100000 */           seg = (addr - 0xc0000) >> 12;           index = (seg >> 3) + 3;           seg &= 7;            /* select 0-7 segments */           return m->fixed_ranges[index*8 + seg];       }   }   /* Match with variable MTRRs. */   for ( seg = 0; seg < num_var_ranges; seg++ )

?? 快捷鍵說明

復制代碼 Ctrl + C
搜索代碼 Ctrl + F
全屏模式 F11
切換主題 Ctrl + Shift + D
顯示快捷鍵 ?
增大字號 Ctrl + =
減小字號 Ctrl + -
亚洲欧美第一页_禁久久精品乱码_粉嫩av一区二区三区免费野_久草精品视频
在线中文字幕一区| 国产成人免费在线视频| 久久精品男人天堂av| 日韩欧美国产三级电影视频| 91精品久久久久久久久99蜜臂| 色婷婷av一区二区三区gif | 91丨porny丨国产入口| 99热99精品| 日韩欧美中文一区二区| bt欧美亚洲午夜电影天堂| 99精品视频一区二区三区| 欧美色综合久久| 国产亚洲精品aa午夜观看| 18欧美乱大交hd1984| 日本一区中文字幕| 成人av资源在线| 91精品国产色综合久久不卡电影| 欧美日韩国产中文| 亚洲精品一线二线三线| 一区二区在线观看免费 | 欧美三级欧美一级| 久久久99精品免费观看不卡| 亚洲综合清纯丝袜自拍| 国产精品中文字幕日韩精品| 欧美日韩成人综合| 国产精品久久影院| 美女视频一区在线观看| 欧美色图激情小说| 国产精品久久久久久亚洲毛片| 久久精品二区亚洲w码| 在线免费一区三区| 综合久久给合久久狠狠狠97色| 久久 天天综合| 欧美日韩另类一区| 中文字幕一区二| 日本不卡中文字幕| 337p亚洲精品色噜噜| 亚洲综合在线观看视频| 色哟哟在线观看一区二区三区| 久久久久国产一区二区三区四区| 毛片不卡一区二区| 日韩亚洲欧美一区二区三区| 亚洲r级在线视频| 精品视频在线看| 日韩精品国产精品| 日韩视频一区在线观看| 欧美aⅴ一区二区三区视频| 欧美日韩国产三级| 日本三级亚洲精品| 国产欧美日韩激情| 在线观看免费视频综合| 亚洲一二三专区| 欧美日韩一级二级三级| 日本成人在线不卡视频| 国产欧美一区二区三区在线看蜜臀 | 久久婷婷成人综合色| 成人黄色网址在线观看| 夜夜嗨av一区二区三区网页| 91精品国产丝袜白色高跟鞋| 欧美一区二区三区视频在线观看| 日韩欧美国产高清| 日产国产欧美视频一区精品 | 国产欧美中文在线| 在线观看亚洲精品| 精彩视频一区二区三区| 亚洲男人电影天堂| 亚洲精品一区二区三区精华液 | 一区二区三区日韩精品视频| 日韩欧美三级在线| 欧洲av一区二区嗯嗯嗯啊| 久久99精品久久久久久动态图 | 色综合网色综合| 国产精品资源在线看| 五月激情综合色| 中文字幕佐山爱一区二区免费| 欧美成人在线直播| 欧美日韩精品二区第二页| 91麻豆自制传媒国产之光| 国产风韵犹存在线视精品| 久久国产综合精品| 美脚の诱脚舐め脚责91| 亚洲精品视频观看| 久久午夜国产精品| 久久午夜色播影院免费高清| 欧美精品一区二区三区很污很色的 | av激情亚洲男人天堂| 99久久久国产精品免费蜜臀| 成人深夜福利app| 91视频在线观看| 91精品91久久久中77777| 99精品久久只有精品| 99精品国产99久久久久久白柏 | 日韩毛片一二三区| 中文字幕中文字幕一区二区| 亚洲视频在线观看三级| 亚洲一区二区欧美日韩| 国产精品久久久久影院色老大| 国产精品女主播av| 亚洲色大成网站www久久九九| 中文字幕第一区综合| 亚洲在线免费播放| 国产在线精品不卡| 91猫先生在线| 久久女同性恋中文字幕| 亚洲欧美日韩国产中文在线| 蜜臀久久99精品久久久久宅男| 不卡视频在线观看| 精品久久久久久无| 亚洲最新在线观看| 韩日av一区二区| 91精品国产综合久久久蜜臀粉嫩 | 97久久精品人人做人人爽| 欧美日韩综合色| 日韩一区欧美二区| 91在线观看免费视频| 久久伊人中文字幕| 日韩高清在线观看| 欧美三级日韩在线| 亚洲欧美日韩一区二区三区在线观看| 日本成人中文字幕在线视频| 欧美日韩亚洲不卡| 亚洲大片在线观看| 欧美在线免费播放| 亚洲高清一区二区三区| 欧美亚洲综合在线| 婷婷一区二区三区| 欧美日韩日本视频| 日本伊人午夜精品| 欧美电影免费观看高清完整版在线观看 | 亚洲男人都懂的| 不卡的av中国片| 亚洲桃色在线一区| www.66久久| 国产精品国产三级国产aⅴ原创| 久久激情五月激情| 韩国女主播一区二区三区| 欧美色老头old∨ideo| 亚洲天堂免费在线观看视频| 亚洲黄色性网站| 国产精品性做久久久久久| 91精品国产色综合久久| 五月天精品一区二区三区| 久久蜜桃av一区精品变态类天堂 | 中文字幕第一区综合| 99这里只有久久精品视频| 日韩欧美国产综合一区 | 精品国产乱码久久久久久牛牛| 欧美日韩精品久久久| 亚洲欧洲国产日本综合| 国产成人免费xxxxxxxx| 91麻豆精品国产91久久久久| 亚洲天堂av老司机| 亚洲一区电影777| 国内不卡的二区三区中文字幕| 欧美伊人精品成人久久综合97| 午夜精品福利久久久| 日韩女同互慰一区二区| 日本乱人伦一区| 极品销魂美女一区二区三区| 国产精品理论在线观看| 欧美精三区欧美精三区| 国产一区二区不卡在线| 亚洲制服丝袜在线| 精品人在线二区三区| 国产欧美在线观看一区| 欧美综合在线视频| 波波电影院一区二区三区| 久久成人精品无人区| 夜夜爽夜夜爽精品视频| 亚洲欧洲av色图| 国产精品久久毛片| 美女视频一区二区| 激情成人午夜视频| 色噜噜久久综合| 成人黄色av网站在线| 欧美96一区二区免费视频| 成人免费在线观看入口| 国产无一区二区| 国产精品福利一区二区| 免费观看91视频大全| 国产精品99久久久久久宅男| 亚洲一区二区三区在线播放| 国产亚洲成aⅴ人片在线观看| 精品欧美一区二区在线观看| 日韩欧美一级特黄在线播放| 日韩国产欧美在线播放| 精品婷婷伊人一区三区三| 国产精品久线在线观看| 精品一区二区三区蜜桃| 91丨九色丨国产丨porny| 丰满白嫩尤物一区二区| av一本久道久久综合久久鬼色| 国产91色综合久久免费分享| 91在线观看高清| 欧美精品日韩综合在线| 日韩精品一区二区三区中文不卡| 国产精品色一区二区三区| 亚洲日本成人在线观看| 一区二区三区中文字幕| 美腿丝袜亚洲色图|