Commit 0f380436 authored by Samuel Pitoiset's avatar Samuel Pitoiset Committed by Ben Skeggs

drm/nouveau/pm: allow to configure domains instead of simple counters

Configuring counters from the userspace require the kernel to handle some
logic related to performance counters. Basically, it has to find a free
slot to assign a counter, to handle extra counting modes like B4/B6 and it
must return and error when it can't configure a counter.

In my opinion, the kernel should not handle all of that logic but it
should only write the configuration sent by the userspace without
checking anything. In other words, it should overwrite the configuration
even if it's already counting and do not return any errors.

This patch allows the userspace to configure a domain instead of
separate counters. This has the advantage to move all of the logic to
the userspace.
Signed-off-by: default avatarSamuel Pitoiset <samuel.pitoiset@gmail.com>
Signed-off-by: default avatarBen Skeggs <bskeggs@redhat.com>
parent 3bfdde17
...@@ -287,34 +287,36 @@ struct nvif_perfmon_query_source_v0 { ...@@ -287,34 +287,36 @@ struct nvif_perfmon_query_source_v0 {
/******************************************************************************* /*******************************************************************************
* perfctr * perfdom
******************************************************************************/ ******************************************************************************/
struct nvif_perfctr_v0 { struct nvif_perfdom_v0 {
__u8 version; __u8 version;
__u8 domain; __u8 domain;
__u8 pad02[2]; __u8 mode;
__u16 logic_op; __u8 pad03[1];
__u8 pad04[2]; struct {
__u8 signal[4]; __u8 signal[4];
__u8 pad06[4]; __u16 logic_op;
} ctr[4];
}; };
#define NVIF_PERFCTR_V0_INIT 0x00 #define NVIF_PERFDOM_V0_INIT 0x00
#define NVIF_PERFCTR_V0_SAMPLE 0x01 #define NVIF_PERFDOM_V0_SAMPLE 0x01
#define NVIF_PERFCTR_V0_READ 0x02 #define NVIF_PERFDOM_V0_READ 0x02
struct nvif_perfctr_init { struct nvif_perfdom_init {
}; };
struct nvif_perfctr_sample { struct nvif_perfdom_sample {
}; };
struct nvif_perfctr_read_v0 { struct nvif_perfdom_read_v0 {
__u8 version; __u8 version;
__u8 pad01[7]; __u8 pad01[7];
__u32 ctr; __u32 ctr[4];
__u32 clk; __u32 clk;
__u8 pad04[4];
}; };
......
...@@ -50,7 +50,7 @@ struct nvif_ioctl_new_v0 { ...@@ -50,7 +50,7 @@ struct nvif_ioctl_new_v0 {
__u32 handle; __u32 handle;
/* these class numbers are made up by us, and not nvidia-assigned */ /* these class numbers are made up by us, and not nvidia-assigned */
#define NVIF_IOCTL_NEW_V0_PERFMON 0x0000ffff #define NVIF_IOCTL_NEW_V0_PERFMON 0x0000ffff
#define NVIF_IOCTL_NEW_V0_PERFCTR 0x0000fffe #define NVIF_IOCTL_NEW_V0_PERFDOM 0x0000fffe
#define NVIF_IOCTL_NEW_V0_CONTROL 0x0000fffd #define NVIF_IOCTL_NEW_V0_CONTROL 0x0000fffd
__u32 oclass; __u32 oclass;
__u8 data[]; /* class data (class.h) */ __u8 data[]; /* class data (class.h) */
......
...@@ -31,9 +31,6 @@ ...@@ -31,9 +31,6 @@
#include <nvif/ioctl.h> #include <nvif/ioctl.h>
#include <nvif/unpack.h> #include <nvif/unpack.h>
#define QUAD_MASK 0x0f
#define QUAD_FREE 0x01
static u8 static u8
nvkm_pm_count_perfdom(struct nvkm_pm *ppm) nvkm_pm_count_perfdom(struct nvkm_pm *ppm)
{ {
...@@ -304,32 +301,27 @@ nvkm_perfmon_ofuncs = { ...@@ -304,32 +301,27 @@ nvkm_perfmon_ofuncs = {
}; };
/******************************************************************************* /*******************************************************************************
* Perfctr object classes * Perfdom object classes
******************************************************************************/ ******************************************************************************/
static int static int
nvkm_perfctr_init(struct nvkm_object *object, void *data, u32 size) nvkm_perfdom_init(struct nvkm_object *object, void *data, u32 size)
{ {
union { union {
struct nvif_perfctr_init none; struct nvif_perfdom_init none;
} *args = data; } *args = data;
struct nvkm_pm *ppm = (void *)object->engine; struct nvkm_pm *ppm = (void *)object->engine;
struct nvkm_perfctr *ctr = (void *)object; struct nvkm_perfdom *dom = (void *)object;
struct nvkm_perfdom *dom = ctr->dom; int ret, i;
int ret;
nv_ioctl(object, "perfctr init size %d\n", size); nv_ioctl(object, "perfdom init size %d\n", size);
if (nvif_unvers(args->none)) { if (nvif_unvers(args->none)) {
nv_ioctl(object, "perfctr init\n"); nv_ioctl(object, "perfdom init\n");
} else } else
return ret; return ret;
ctr->slot = ffs(dom->quad) - 1; for (i = 0; i < 4; i++)
if (ctr->slot < 0) { if (dom->ctr[i])
/* no free slots are available */ dom->func->init(ppm, dom, dom->ctr[i]);
return -EINVAL;
}
dom->quad &= ~(QUAD_FREE << ctr->slot);
dom->func->init(ppm, dom, ctr);
/* start next batch of counters for sampling */ /* start next batch of counters for sampling */
dom->func->next(ppm, dom); dom->func->next(ppm, dom);
...@@ -337,74 +329,70 @@ nvkm_perfctr_init(struct nvkm_object *object, void *data, u32 size) ...@@ -337,74 +329,70 @@ nvkm_perfctr_init(struct nvkm_object *object, void *data, u32 size)
} }
static int static int
nvkm_perfctr_sample(struct nvkm_object *object, void *data, u32 size) nvkm_perfdom_sample(struct nvkm_object *object, void *data, u32 size)
{ {
union { union {
struct nvif_perfctr_sample none; struct nvif_perfdom_sample none;
} *args = data; } *args = data;
struct nvkm_pm *ppm = (void *)object->engine; struct nvkm_pm *ppm = (void *)object->engine;
struct nvkm_perfctr *ctr;
struct nvkm_perfdom *dom; struct nvkm_perfdom *dom;
int ret; int ret;
nv_ioctl(object, "perfctr sample size %d\n", size); nv_ioctl(object, "perfdom sample size %d\n", size);
if (nvif_unvers(args->none)) { if (nvif_unvers(args->none)) {
nv_ioctl(object, "perfctr sample\n"); nv_ioctl(object, "perfdom sample\n");
} else } else
return ret; return ret;
ppm->sequence++; ppm->sequence++;
list_for_each_entry(dom, &ppm->domains, head) { /* sample previous batch of counters */
/* sample previous batch of counters */ list_for_each_entry(dom, &ppm->domains, head)
if (dom->quad != QUAD_MASK) { dom->func->next(ppm, dom);
dom->func->next(ppm, dom);
/* read counter values */
list_for_each_entry(ctr, &dom->list, head) {
dom->func->read(ppm, dom, ctr);
ctr->slot = -1;
}
dom->quad = QUAD_MASK;
}
}
return 0; return 0;
} }
static int static int
nvkm_perfctr_read(struct nvkm_object *object, void *data, u32 size) nvkm_perfdom_read(struct nvkm_object *object, void *data, u32 size)
{ {
union { union {
struct nvif_perfctr_read_v0 v0; struct nvif_perfdom_read_v0 v0;
} *args = data; } *args = data;
struct nvkm_perfctr *ctr = (void *)object; struct nvkm_pm *ppm = (void *)object->engine;
int ret; struct nvkm_perfdom *dom = (void *)object;
int ret, i;
nv_ioctl(object, "perfctr read size %d\n", size); nv_ioctl(object, "perfdom read size %d\n", size);
if (nvif_unpack(args->v0, 0, 0, false)) { if (nvif_unpack(args->v0, 0, 0, false)) {
nv_ioctl(object, "perfctr read vers %d\n", args->v0.version); nv_ioctl(object, "perfdom read vers %d\n", args->v0.version);
} else } else
return ret; return ret;
if (!ctr->clk) for (i = 0; i < 4; i++) {
if (dom->ctr[i])
dom->func->read(ppm, dom, dom->ctr[i]);
}
if (!dom->clk)
return -EAGAIN; return -EAGAIN;
args->v0.clk = ctr->clk; for (i = 0; i < 4; i++)
args->v0.ctr = ctr->ctr; if (dom->ctr[i])
args->v0.ctr[i] = dom->ctr[i]->ctr;
args->v0.clk = dom->clk;
return 0; return 0;
} }
static int static int
nvkm_perfctr_mthd(struct nvkm_object *object, u32 mthd, void *data, u32 size) nvkm_perfdom_mthd(struct nvkm_object *object, u32 mthd, void *data, u32 size)
{ {
switch (mthd) { switch (mthd) {
case NVIF_PERFCTR_V0_INIT: case NVIF_PERFDOM_V0_INIT:
return nvkm_perfctr_init(object, data, size); return nvkm_perfdom_init(object, data, size);
case NVIF_PERFCTR_V0_SAMPLE: case NVIF_PERFDOM_V0_SAMPLE:
return nvkm_perfctr_sample(object, data, size); return nvkm_perfdom_sample(object, data, size);
case NVIF_PERFCTR_V0_READ: case NVIF_PERFDOM_V0_READ:
return nvkm_perfctr_read(object, data, size); return nvkm_perfdom_read(object, data, size);
default: default:
break; break;
} }
...@@ -412,70 +400,107 @@ nvkm_perfctr_mthd(struct nvkm_object *object, u32 mthd, void *data, u32 size) ...@@ -412,70 +400,107 @@ nvkm_perfctr_mthd(struct nvkm_object *object, u32 mthd, void *data, u32 size)
} }
static void static void
nvkm_perfctr_dtor(struct nvkm_object *object) nvkm_perfdom_dtor(struct nvkm_object *object)
{ {
struct nvkm_perfctr *ctr = (void *)object; struct nvkm_perfdom *dom = (void *)object;
if (ctr->dom) int i;
ctr->dom->quad |= (QUAD_FREE << ctr->slot);
if (ctr->head.next) for (i = 0; i < 4; i++) {
list_del(&ctr->head); struct nvkm_perfctr *ctr = dom->ctr[i];
nvkm_object_destroy(&ctr->base); if (ctr && ctr->head.next)
list_del(&ctr->head);
kfree(ctr);
}
nvkm_object_destroy(&dom->base);
} }
static int static int
nvkm_perfctr_ctor(struct nvkm_object *parent, struct nvkm_object *engine, nvkm_perfctr_new(struct nvkm_perfdom *dom, int slot,
struct nvkm_perfsig *signal[4], uint16_t logic_op,
struct nvkm_perfctr **pctr)
{
struct nvkm_perfctr *ctr;
int i;
if (!dom)
return -EINVAL;
ctr = *pctr = kzalloc(sizeof(*ctr), GFP_KERNEL);
if (!ctr)
return -ENOMEM;
ctr->logic_op = logic_op;
ctr->slot = slot;
for (i = 0; i < 4; i++) {
if (signal[i])
ctr->signal[i] = signal[i] - dom->signal;
}
list_add_tail(&ctr->head, &dom->list);
return 0;
}
static int
nvkm_perfdom_ctor(struct nvkm_object *parent, struct nvkm_object *engine,
struct nvkm_oclass *oclass, void *data, u32 size, struct nvkm_oclass *oclass, void *data, u32 size,
struct nvkm_object **pobject) struct nvkm_object **pobject)
{ {
union { union {
struct nvif_perfctr_v0 v0; struct nvif_perfdom_v0 v0;
} *args = data; } *args = data;
struct nvkm_pm *ppm = (void *)engine; struct nvkm_pm *ppm = (void *)engine;
struct nvkm_perfdom *dom = NULL; struct nvkm_perfdom *sdom = NULL;
struct nvkm_perfsig *sig[4] = {}; struct nvkm_perfctr *ctr[4] = {};
struct nvkm_perfctr *ctr; struct nvkm_perfdom *dom;
int ret, i; int c, s;
int ret;
nv_ioctl(parent, "create perfctr size %d\n", size); nv_ioctl(parent, "create perfdom size %d\n", size);
if (nvif_unpack(args->v0, 0, 0, false)) { if (nvif_unpack(args->v0, 0, 0, false)) {
nv_ioctl(parent, "create perfctr vers %d logic_op %04x\n", nv_ioctl(parent, "create perfdom vers %d dom %d mode %02x\n",
args->v0.version, args->v0.logic_op); args->v0.version, args->v0.domain, args->v0.mode);
} else } else
return ret; return ret;
for (i = 0; i < ARRAY_SIZE(args->v0.signal); i++) { for (c = 0; c < ARRAY_SIZE(args->v0.ctr); c++) {
sig[i] = nvkm_perfsig_find(ppm, args->v0.domain, struct nvkm_perfsig *sig[4] = {};
args->v0.signal[i], &dom); for (s = 0; s < ARRAY_SIZE(args->v0.ctr[c].signal); s++) {
if (args->v0.signal[i] && !sig[i]) sig[s] = nvkm_perfsig_find(ppm, args->v0.domain,
return -EINVAL; args->v0.ctr[c].signal[s],
&sdom);
if (args->v0.ctr[c].signal[s] && !sig[s])
return -EINVAL;
}
ret = nvkm_perfctr_new(sdom, c, sig,
args->v0.ctr[c].logic_op, &ctr[c]);
if (ret)
return ret;
} }
if (!dom) if (!sdom)
return -EINVAL; return -EINVAL;
ret = nvkm_object_create(parent, engine, oclass, 0, &ctr); ret = nvkm_object_create(parent, engine, oclass, 0, &dom);
*pobject = nv_object(ctr); *pobject = nv_object(dom);
if (ret) if (ret)
return ret; return ret;
ctr->dom = dom; dom->func = sdom->func;
ctr->slot = -1; dom->addr = sdom->addr;
ctr->logic_op = args->v0.logic_op; dom->mode = args->v0.mode;
ctr->signal[0] = sig[0]; for (c = 0; c < ARRAY_SIZE(ctr); c++)
ctr->signal[1] = sig[1]; dom->ctr[c] = ctr[c];
ctr->signal[2] = sig[2];
ctr->signal[3] = sig[3];
list_add_tail(&ctr->head, &dom->list);
return 0; return 0;
} }
static struct nvkm_ofuncs static struct nvkm_ofuncs
nvkm_perfctr_ofuncs = { nvkm_perfdom_ofuncs = {
.ctor = nvkm_perfctr_ctor, .ctor = nvkm_perfdom_ctor,
.dtor = nvkm_perfctr_dtor, .dtor = nvkm_perfdom_dtor,
.init = nvkm_object_init, .init = nvkm_object_init,
.fini = nvkm_object_fini, .fini = nvkm_object_fini,
.mthd = nvkm_perfctr_mthd, .mthd = nvkm_perfdom_mthd,
}; };
struct nvkm_oclass struct nvkm_oclass
...@@ -484,8 +509,8 @@ nvkm_pm_sclass[] = { ...@@ -484,8 +509,8 @@ nvkm_pm_sclass[] = {
.handle = NVIF_IOCTL_NEW_V0_PERFMON, .handle = NVIF_IOCTL_NEW_V0_PERFMON,
.ofuncs = &nvkm_perfmon_ofuncs, .ofuncs = &nvkm_perfmon_ofuncs,
}, },
{ .handle = NVIF_IOCTL_NEW_V0_PERFCTR, { .handle = NVIF_IOCTL_NEW_V0_PERFDOM,
.ofuncs = &nvkm_perfctr_ofuncs, .ofuncs = &nvkm_perfdom_ofuncs,
}, },
{}, {},
}; };
...@@ -640,7 +665,6 @@ nvkm_perfdom_new(struct nvkm_pm *ppm, const char *name, u32 mask, ...@@ -640,7 +665,6 @@ nvkm_perfdom_new(struct nvkm_pm *ppm, const char *name, u32 mask,
INIT_LIST_HEAD(&dom->list); INIT_LIST_HEAD(&dom->list);
dom->func = sdom->func; dom->func = sdom->func;
dom->addr = addr; dom->addr = addr;
dom->quad = QUAD_MASK;
dom->signal_nr = sdom->signal_nr; dom->signal_nr = sdom->signal_nr;
ssig = (sdom++)->signal; ssig = (sdom++)->signal;
......
...@@ -48,12 +48,10 @@ gf100_perfctr_init(struct nvkm_pm *ppm, struct nvkm_perfdom *dom, ...@@ -48,12 +48,10 @@ gf100_perfctr_init(struct nvkm_pm *ppm, struct nvkm_perfdom *dom,
u32 src = 0x00000000; u32 src = 0x00000000;
int i; int i;
for (i = 0; i < 4; i++) { for (i = 0; i < 4; i++)
if (ctr->signal[i]) src |= ctr->signal[i] << (i * 8);
src |= (ctr->signal[i] - dom->signal) << (i * 8);
}
nv_wr32(priv, dom->addr + 0x09c, 0x00040002); nv_wr32(priv, dom->addr + 0x09c, 0x00040002 | (dom->mode << 3));
nv_wr32(priv, dom->addr + 0x100, 0x00000000); nv_wr32(priv, dom->addr + 0x100, 0x00000000);
nv_wr32(priv, dom->addr + 0x040 + (cntr->base.slot * 0x08), src); nv_wr32(priv, dom->addr + 0x040 + (cntr->base.slot * 0x08), src);
nv_wr32(priv, dom->addr + 0x044 + (cntr->base.slot * 0x08), log); nv_wr32(priv, dom->addr + 0x044 + (cntr->base.slot * 0x08), log);
...@@ -72,7 +70,7 @@ gf100_perfctr_read(struct nvkm_pm *ppm, struct nvkm_perfdom *dom, ...@@ -72,7 +70,7 @@ gf100_perfctr_read(struct nvkm_pm *ppm, struct nvkm_perfdom *dom,
case 2: cntr->base.ctr = nv_rd32(priv, dom->addr + 0x080); break; case 2: cntr->base.ctr = nv_rd32(priv, dom->addr + 0x080); break;
case 3: cntr->base.ctr = nv_rd32(priv, dom->addr + 0x090); break; case 3: cntr->base.ctr = nv_rd32(priv, dom->addr + 0x090); break;
} }
cntr->base.clk = nv_rd32(priv, dom->addr + 0x070); dom->clk = nv_rd32(priv, dom->addr + 0x070);
} }
static void static void
......
...@@ -33,12 +33,10 @@ nv40_perfctr_init(struct nvkm_pm *ppm, struct nvkm_perfdom *dom, ...@@ -33,12 +33,10 @@ nv40_perfctr_init(struct nvkm_pm *ppm, struct nvkm_perfdom *dom,
u32 src = 0x00000000; u32 src = 0x00000000;
int i; int i;
for (i = 0; i < 4; i++) { for (i = 0; i < 4; i++)
if (ctr->signal[i]) src |= ctr->signal[i] << (i * 8);
src |= (ctr->signal[i] - dom->signal) << (i * 8);
}
nv_wr32(priv, 0x00a7c0 + dom->addr, 0x00000001); nv_wr32(priv, 0x00a7c0 + dom->addr, 0x00000001 | (dom->mode << 4));
nv_wr32(priv, 0x00a400 + dom->addr + (cntr->base.slot * 0x40), src); nv_wr32(priv, 0x00a400 + dom->addr + (cntr->base.slot * 0x40), src);
nv_wr32(priv, 0x00a420 + dom->addr + (cntr->base.slot * 0x40), log); nv_wr32(priv, 0x00a420 + dom->addr + (cntr->base.slot * 0x40), log);
} }
...@@ -56,7 +54,7 @@ nv40_perfctr_read(struct nvkm_pm *ppm, struct nvkm_perfdom *dom, ...@@ -56,7 +54,7 @@ nv40_perfctr_read(struct nvkm_pm *ppm, struct nvkm_perfdom *dom,
case 2: cntr->base.ctr = nv_rd32(priv, 0x00a680 + dom->addr); break; case 2: cntr->base.ctr = nv_rd32(priv, 0x00a680 + dom->addr); break;
case 3: cntr->base.ctr = nv_rd32(priv, 0x00a740 + dom->addr); break; case 3: cntr->base.ctr = nv_rd32(priv, 0x00a740 + dom->addr); break;
} }
cntr->base.clk = nv_rd32(priv, 0x00a600 + dom->addr); dom->clk = nv_rd32(priv, 0x00a600 + dom->addr);
} }
static void static void
......
...@@ -3,13 +3,10 @@ ...@@ -3,13 +3,10 @@
#include <engine/pm.h> #include <engine/pm.h>
struct nvkm_perfctr { struct nvkm_perfctr {
struct nvkm_object base;
struct list_head head; struct list_head head;
struct nvkm_perfsig *signal[4]; u8 signal[4];
struct nvkm_perfdom *dom;
int slot; int slot;
u32 logic_op; u32 logic_op;
u32 clk;
u32 ctr; u32 ctr;
}; };
...@@ -63,12 +60,15 @@ struct nvkm_specdom { ...@@ -63,12 +60,15 @@ struct nvkm_specdom {
}; };
struct nvkm_perfdom { struct nvkm_perfdom {
struct nvkm_object base;
struct list_head head; struct list_head head;
struct list_head list; struct list_head list;
const struct nvkm_funcdom *func; const struct nvkm_funcdom *func;
struct nvkm_perfctr *ctr[4];
char name[32]; char name[32];
u32 addr; u32 addr;
u8 quad; u8 mode;
u32 clk;
u16 signal_nr; u16 signal_nr;
struct nvkm_perfsig signal[]; struct nvkm_perfsig signal[];
}; };
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment