1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
|
/*
* Copyright 2017 Advanced Micro Devices, Inc.
*
* Permission is hereby granted, free of charge, to any person obtaining a
* copy of this software and associated documentation files (the "Software"),
* to deal in the Software without restriction, including without limitation
* the rights to use, copy, modify, merge, publish, distribute, sublicense,
* and/or sell copies of the Software, and to permit persons to whom the
* Software is furnished to do so, subject to the following conditions:
*
* The above copyright notice and this permission notice shall be included in
* all copies or substantial portions of the Software.
*
* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
* IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
* FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL
* THE COPYRIGHT HOLDER(S) OR AUTHOR(S) BE LIABLE FOR ANY CLAIM, DAMAGES OR
* OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE,
* ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR
* OTHER DEALINGS IN THE SOFTWARE.
*
* Authors: Tom St Denis <tom.stdenis@amd.com>
*
*/
#include "umrapp.h"
#include <inttypes.h>
/* set a register based on regpath e.g.
*
* stoney.uvd6.mmFOO.bit
*/
int umr_set_register_bit(struct umr_asic *asic, char *regpath, char *regvalue)
{
char rpath[512];
int i, j, k, fd;
uint32_t value, mask, copy;
uint64_t addr, scale;
/* scan until we compare with regpath... */
for (i = 0; i < asic->no_blocks; i++) {
if (!memcmp(regpath, "*.", 2))
snprintf(rpath, sizeof(rpath)-1, "*.%s.", asic->blocks[i]->ipname);
else
snprintf(rpath, sizeof(rpath)-1, "%s.%s.", asic->asicname, asic->blocks[i]->ipname);
if (memcmp(regpath, rpath, strlen(rpath)) == 0) {
for (j = 0; j < asic->blocks[i]->no_regs; j++) {
if (asic->blocks[i]->regs[j].bits) {
for (k = 0; k < asic->blocks[i]->regs[j].no_bits; k++) {
if (!memcmp(regpath, "*.", 2))
snprintf(rpath, sizeof(rpath)-1, "*.%s.%s.%s", asic->blocks[i]->ipname, asic->blocks[i]->regs[j].regname, asic->blocks[i]->regs[j].bits[k].regname);
else
snprintf(rpath, sizeof(rpath)-1, "%s.%s.%s.%s", asic->asicname, asic->blocks[i]->ipname, asic->blocks[i]->regs[j].regname, asic->blocks[i]->regs[j].bits[k].regname);
if (!strcmp(regpath, rpath)) {
sscanf(regvalue, "%"SCNx32, &value);
mask = (1UL<<((1+asic->blocks[i]->regs[j].bits[k].stop)-asic->blocks[i]->regs[j].bits[k].start))-1;
mask <<= asic->blocks[i]->regs[j].bits[k].start;
if (asic->pci.mem == NULL) {
// set this register
switch (asic->blocks[i]->regs[j].type){
case REG_MMIO: fd = asic->fd.mmio; scale = 4; break;
case REG_DIDT: fd = asic->fd.didt; scale = 1; break;
case REG_PCIE: fd = asic->fd.pcie; scale = 1; break;
case REG_SMC: fd = asic->fd.smc; scale = 4; break;
default: return -1;
}
if (asic->blocks[i]->grant) {
if (asic->blocks[i]->grant(asic)) {
return -1;
}
}
if (options.use_bank && asic->blocks[i]->regs[j].type == REG_MMIO)
addr =
(1ULL << 62) |
(((uint64_t)options.se_bank) << 24) |
(((uint64_t)options.sh_bank) << 34) |
(((uint64_t)options.instance_bank) << 44);
else
addr = 0;
lseek(fd, addr | (asic->blocks[i]->regs[j].addr*scale), SEEK_SET);
read(fd, ©, 4);
// read-modify-write value back
copy &= ~mask;
value = (value << asic->blocks[i]->regs[j].bits[k].start) & mask;
copy |= value;
lseek(fd, addr | (asic->blocks[i]->regs[j].addr<<2), SEEK_SET);
write(fd, ©, 4);
if (!options.quiet) printf("%s <= 0x%08lx\n", regpath, (unsigned long)copy);
if (asic->blocks[i]->release) {
if (asic->blocks[i]->release(asic)) {
return -1;
}
}
} else if (asic->blocks[i]->regs[j].type == REG_MMIO) {
copy = asic->pci.mem[asic->blocks[i]->regs[j].addr] & ~mask;
copy |= (value << asic->blocks[i]->regs[j].bits[k].start) & mask;
asic->pci.mem[asic->blocks[i]->regs[j].addr] = copy;
if (!options.quiet) printf("%s <= 0x%08lx\n", regpath, (unsigned long)copy);
}
return 0;
}
}
}
}
}
}
fprintf(stderr, "Path <%s> not found on this ASIC\n", regpath);
return -1;
}
|