|
@@ -38,7 +38,7 @@ static guint smmu_iotlb_key_hash(gconstpointer v)
|
|
|
|
|
|
/* Jenkins hash */
|
|
/* Jenkins hash */
|
|
a = b = c = JHASH_INITVAL + sizeof(*key);
|
|
a = b = c = JHASH_INITVAL + sizeof(*key);
|
|
- a += key->asid + key->level + key->tg;
|
|
|
|
|
|
+ a += key->asid + key->vmid + key->level + key->tg;
|
|
b += extract64(key->iova, 0, 32);
|
|
b += extract64(key->iova, 0, 32);
|
|
c += extract64(key->iova, 32, 32);
|
|
c += extract64(key->iova, 32, 32);
|
|
|
|
|
|
@@ -53,13 +53,15 @@ static gboolean smmu_iotlb_key_equal(gconstpointer v1, gconstpointer v2)
|
|
SMMUIOTLBKey *k1 = (SMMUIOTLBKey *)v1, *k2 = (SMMUIOTLBKey *)v2;
|
|
SMMUIOTLBKey *k1 = (SMMUIOTLBKey *)v1, *k2 = (SMMUIOTLBKey *)v2;
|
|
|
|
|
|
return (k1->asid == k2->asid) && (k1->iova == k2->iova) &&
|
|
return (k1->asid == k2->asid) && (k1->iova == k2->iova) &&
|
|
- (k1->level == k2->level) && (k1->tg == k2->tg);
|
|
|
|
|
|
+ (k1->level == k2->level) && (k1->tg == k2->tg) &&
|
|
|
|
+ (k1->vmid == k2->vmid);
|
|
}
|
|
}
|
|
|
|
|
|
-SMMUIOTLBKey smmu_get_iotlb_key(uint16_t asid, uint64_t iova,
|
|
|
|
|
|
+SMMUIOTLBKey smmu_get_iotlb_key(uint16_t asid, uint16_t vmid, uint64_t iova,
|
|
uint8_t tg, uint8_t level)
|
|
uint8_t tg, uint8_t level)
|
|
{
|
|
{
|
|
- SMMUIOTLBKey key = {.asid = asid, .iova = iova, .tg = tg, .level = level};
|
|
|
|
|
|
+ SMMUIOTLBKey key = {.asid = asid, .vmid = vmid, .iova = iova,
|
|
|
|
+ .tg = tg, .level = level};
|
|
|
|
|
|
return key;
|
|
return key;
|
|
}
|
|
}
|
|
@@ -78,7 +80,8 @@ SMMUTLBEntry *smmu_iotlb_lookup(SMMUState *bs, SMMUTransCfg *cfg,
|
|
uint64_t mask = subpage_size - 1;
|
|
uint64_t mask = subpage_size - 1;
|
|
SMMUIOTLBKey key;
|
|
SMMUIOTLBKey key;
|
|
|
|
|
|
- key = smmu_get_iotlb_key(cfg->asid, iova & ~mask, tg, level);
|
|
|
|
|
|
+ key = smmu_get_iotlb_key(cfg->asid, cfg->s2cfg.vmid,
|
|
|
|
+ iova & ~mask, tg, level);
|
|
entry = g_hash_table_lookup(bs->iotlb, &key);
|
|
entry = g_hash_table_lookup(bs->iotlb, &key);
|
|
if (entry) {
|
|
if (entry) {
|
|
break;
|
|
break;
|
|
@@ -88,13 +91,13 @@ SMMUTLBEntry *smmu_iotlb_lookup(SMMUState *bs, SMMUTransCfg *cfg,
|
|
|
|
|
|
if (entry) {
|
|
if (entry) {
|
|
cfg->iotlb_hits++;
|
|
cfg->iotlb_hits++;
|
|
- trace_smmu_iotlb_lookup_hit(cfg->asid, iova,
|
|
|
|
|
|
+ trace_smmu_iotlb_lookup_hit(cfg->asid, cfg->s2cfg.vmid, iova,
|
|
cfg->iotlb_hits, cfg->iotlb_misses,
|
|
cfg->iotlb_hits, cfg->iotlb_misses,
|
|
100 * cfg->iotlb_hits /
|
|
100 * cfg->iotlb_hits /
|
|
(cfg->iotlb_hits + cfg->iotlb_misses));
|
|
(cfg->iotlb_hits + cfg->iotlb_misses));
|
|
} else {
|
|
} else {
|
|
cfg->iotlb_misses++;
|
|
cfg->iotlb_misses++;
|
|
- trace_smmu_iotlb_lookup_miss(cfg->asid, iova,
|
|
|
|
|
|
+ trace_smmu_iotlb_lookup_miss(cfg->asid, cfg->s2cfg.vmid, iova,
|
|
cfg->iotlb_hits, cfg->iotlb_misses,
|
|
cfg->iotlb_hits, cfg->iotlb_misses,
|
|
100 * cfg->iotlb_hits /
|
|
100 * cfg->iotlb_hits /
|
|
(cfg->iotlb_hits + cfg->iotlb_misses));
|
|
(cfg->iotlb_hits + cfg->iotlb_misses));
|
|
@@ -111,8 +114,10 @@ void smmu_iotlb_insert(SMMUState *bs, SMMUTransCfg *cfg, SMMUTLBEntry *new)
|
|
smmu_iotlb_inv_all(bs);
|
|
smmu_iotlb_inv_all(bs);
|
|
}
|
|
}
|
|
|
|
|
|
- *key = smmu_get_iotlb_key(cfg->asid, new->entry.iova, tg, new->level);
|
|
|
|
- trace_smmu_iotlb_insert(cfg->asid, new->entry.iova, tg, new->level);
|
|
|
|
|
|
+ *key = smmu_get_iotlb_key(cfg->asid, cfg->s2cfg.vmid, new->entry.iova,
|
|
|
|
+ tg, new->level);
|
|
|
|
+ trace_smmu_iotlb_insert(cfg->asid, cfg->s2cfg.vmid, new->entry.iova,
|
|
|
|
+ tg, new->level);
|
|
g_hash_table_insert(bs->iotlb, key, new);
|
|
g_hash_table_insert(bs->iotlb, key, new);
|
|
}
|
|
}
|
|
|
|
|
|
@@ -130,8 +135,7 @@ static gboolean smmu_hash_remove_by_asid(gpointer key, gpointer value,
|
|
|
|
|
|
return SMMU_IOTLB_ASID(*iotlb_key) == asid;
|
|
return SMMU_IOTLB_ASID(*iotlb_key) == asid;
|
|
}
|
|
}
|
|
-
|
|
|
|
-static gboolean smmu_hash_remove_by_asid_iova(gpointer key, gpointer value,
|
|
|
|
|
|
+static gboolean smmu_hash_remove_by_asid_vmid_iova(gpointer key, gpointer value,
|
|
gpointer user_data)
|
|
gpointer user_data)
|
|
{
|
|
{
|
|
SMMUTLBEntry *iter = (SMMUTLBEntry *)value;
|
|
SMMUTLBEntry *iter = (SMMUTLBEntry *)value;
|
|
@@ -142,18 +146,21 @@ static gboolean smmu_hash_remove_by_asid_iova(gpointer key, gpointer value,
|
|
if (info->asid >= 0 && info->asid != SMMU_IOTLB_ASID(iotlb_key)) {
|
|
if (info->asid >= 0 && info->asid != SMMU_IOTLB_ASID(iotlb_key)) {
|
|
return false;
|
|
return false;
|
|
}
|
|
}
|
|
|
|
+ if (info->vmid >= 0 && info->vmid != SMMU_IOTLB_VMID(iotlb_key)) {
|
|
|
|
+ return false;
|
|
|
|
+ }
|
|
return ((info->iova & ~entry->addr_mask) == entry->iova) ||
|
|
return ((info->iova & ~entry->addr_mask) == entry->iova) ||
|
|
((entry->iova & ~info->mask) == info->iova);
|
|
((entry->iova & ~info->mask) == info->iova);
|
|
}
|
|
}
|
|
|
|
|
|
-void smmu_iotlb_inv_iova(SMMUState *s, int asid, dma_addr_t iova,
|
|
|
|
|
|
+void smmu_iotlb_inv_iova(SMMUState *s, int asid, int vmid, dma_addr_t iova,
|
|
uint8_t tg, uint64_t num_pages, uint8_t ttl)
|
|
uint8_t tg, uint64_t num_pages, uint8_t ttl)
|
|
{
|
|
{
|
|
/* if tg is not set we use 4KB range invalidation */
|
|
/* if tg is not set we use 4KB range invalidation */
|
|
uint8_t granule = tg ? tg * 2 + 10 : 12;
|
|
uint8_t granule = tg ? tg * 2 + 10 : 12;
|
|
|
|
|
|
if (ttl && (num_pages == 1) && (asid >= 0)) {
|
|
if (ttl && (num_pages == 1) && (asid >= 0)) {
|
|
- SMMUIOTLBKey key = smmu_get_iotlb_key(asid, iova, tg, ttl);
|
|
|
|
|
|
+ SMMUIOTLBKey key = smmu_get_iotlb_key(asid, vmid, iova, tg, ttl);
|
|
|
|
|
|
if (g_hash_table_remove(s->iotlb, &key)) {
|
|
if (g_hash_table_remove(s->iotlb, &key)) {
|
|
return;
|
|
return;
|
|
@@ -166,10 +173,11 @@ void smmu_iotlb_inv_iova(SMMUState *s, int asid, dma_addr_t iova,
|
|
|
|
|
|
SMMUIOTLBPageInvInfo info = {
|
|
SMMUIOTLBPageInvInfo info = {
|
|
.asid = asid, .iova = iova,
|
|
.asid = asid, .iova = iova,
|
|
|
|
+ .vmid = vmid,
|
|
.mask = (num_pages * 1 << granule) - 1};
|
|
.mask = (num_pages * 1 << granule) - 1};
|
|
|
|
|
|
g_hash_table_foreach_remove(s->iotlb,
|
|
g_hash_table_foreach_remove(s->iotlb,
|
|
- smmu_hash_remove_by_asid_iova,
|
|
|
|
|
|
+ smmu_hash_remove_by_asid_vmid_iova,
|
|
&info);
|
|
&info);
|
|
}
|
|
}
|
|
|
|
|