mirror of
https://gitlab.freedesktop.org/mesa/drm.git
synced 2025-12-20 23:20:11 +01:00
Merge branch 'modesetting-gem' of ssh://git.freedesktop.org/git/mesa/drm into modesetting-gem
This commit is contained in:
commit
c67a83dfe1
12 changed files with 377 additions and 41 deletions
|
|
@ -36,11 +36,13 @@ libdrm_radeon_la_LIBADD = ../libdrm.la @PTHREADSTUBS_LIBS@
|
||||||
|
|
||||||
libdrm_radeon_la_SOURCES = \
|
libdrm_radeon_la_SOURCES = \
|
||||||
radeon_bo_gem.c \
|
radeon_bo_gem.c \
|
||||||
radeon_cs_gem.c
|
radeon_cs_gem.c \
|
||||||
|
radeon_track.c
|
||||||
|
|
||||||
libdrm_radeonincludedir = ${includedir}/drm
|
libdrm_radeonincludedir = ${includedir}/drm
|
||||||
libdrm_radeoninclude_HEADERS = \
|
libdrm_radeoninclude_HEADERS = \
|
||||||
radeon_bo.h \
|
radeon_bo.h \
|
||||||
radeon_cs.h \
|
radeon_cs.h \
|
||||||
radeon_bo_gem.h \
|
radeon_bo_gem.h \
|
||||||
radeon_cs_gem.h
|
radeon_cs_gem.h \
|
||||||
|
radeon_track.h
|
||||||
|
|
|
||||||
|
|
@ -32,6 +32,7 @@
|
||||||
|
|
||||||
#include <stdio.h>
|
#include <stdio.h>
|
||||||
#include <stdint.h>
|
#include <stdint.h>
|
||||||
|
#include "radeon_track.h"
|
||||||
|
|
||||||
/* bo object */
|
/* bo object */
|
||||||
#define RADEON_BO_FLAGS_MACRO_TILE 1
|
#define RADEON_BO_FLAGS_MACRO_TILE 1
|
||||||
|
|
@ -46,6 +47,9 @@ struct radeon_bo {
|
||||||
uint32_t domains;
|
uint32_t domains;
|
||||||
uint32_t flags;
|
uint32_t flags;
|
||||||
unsigned cref;
|
unsigned cref;
|
||||||
|
#ifdef RADEON_BO_TRACK
|
||||||
|
struct radeon_track *track;
|
||||||
|
#endif
|
||||||
void *ptr;
|
void *ptr;
|
||||||
struct radeon_bo_manager *bom;
|
struct radeon_bo_manager *bom;
|
||||||
};
|
};
|
||||||
|
|
@ -59,7 +63,7 @@ struct radeon_bo_funcs {
|
||||||
uint32_t domains,
|
uint32_t domains,
|
||||||
uint32_t flags);
|
uint32_t flags);
|
||||||
void (*bo_ref)(struct radeon_bo *bo);
|
void (*bo_ref)(struct radeon_bo *bo);
|
||||||
void (*bo_unref)(struct radeon_bo *bo);
|
struct radeon_bo *(*bo_unref)(struct radeon_bo *bo);
|
||||||
int (*bo_map)(struct radeon_bo *bo, int write);
|
int (*bo_map)(struct radeon_bo *bo, int write);
|
||||||
int (*bo_unmap)(struct radeon_bo *bo);
|
int (*bo_unmap)(struct radeon_bo *bo);
|
||||||
};
|
};
|
||||||
|
|
@ -67,16 +71,17 @@ struct radeon_bo_funcs {
|
||||||
struct radeon_bo_manager {
|
struct radeon_bo_manager {
|
||||||
struct radeon_bo_funcs *funcs;
|
struct radeon_bo_funcs *funcs;
|
||||||
int fd;
|
int fd;
|
||||||
|
struct radeon_tracker tracker;
|
||||||
};
|
};
|
||||||
|
|
||||||
static inline void _radeon_bo_debug(struct radeon_bo *bo,
|
static inline void _radeon_bo_debug(struct radeon_bo *bo,
|
||||||
int opcode,
|
const char *op,
|
||||||
const char *file,
|
const char *file,
|
||||||
const char *func,
|
const char *func,
|
||||||
int line)
|
int line)
|
||||||
{
|
{
|
||||||
fprintf(stderr, "%02d %p 0x%08X 0x%08X 0x%08X [%s %s %d]\n",
|
fprintf(stderr, "%s %p 0x%08X 0x%08X 0x%08X [%s %s %d]\n",
|
||||||
opcode, bo, bo->handle, bo->size, bo->cref, file, func, line);
|
op, bo, bo->handle, bo->size, bo->cref, file, func, line);
|
||||||
}
|
}
|
||||||
|
|
||||||
static inline struct radeon_bo *_radeon_bo_open(struct radeon_bo_manager *bom,
|
static inline struct radeon_bo *_radeon_bo_open(struct radeon_bo_manager *bom,
|
||||||
|
|
@ -90,10 +95,12 @@ static inline struct radeon_bo *_radeon_bo_open(struct radeon_bo_manager *bom,
|
||||||
int line)
|
int line)
|
||||||
{
|
{
|
||||||
struct radeon_bo *bo;
|
struct radeon_bo *bo;
|
||||||
|
|
||||||
bo = bom->funcs->bo_open(bom, handle, size, alignment, domains, flags);
|
bo = bom->funcs->bo_open(bom, handle, size, alignment, domains, flags);
|
||||||
#ifdef RADEON_BO_TRACK_OPEN
|
#ifdef RADEON_BO_TRACK
|
||||||
if (bo) {
|
if (bo) {
|
||||||
_radeon_bo_debug(bo, 1, file, func, line);
|
bo->track = radeon_tracker_add_track(&bom->tracker, bo->handle);
|
||||||
|
radeon_track_add_event(bo->track, file, func, "open", line);
|
||||||
}
|
}
|
||||||
#endif
|
#endif
|
||||||
return bo;
|
return bo;
|
||||||
|
|
@ -105,22 +112,26 @@ static inline void _radeon_bo_ref(struct radeon_bo *bo,
|
||||||
int line)
|
int line)
|
||||||
{
|
{
|
||||||
bo->cref++;
|
bo->cref++;
|
||||||
#ifdef RADEON_BO_TRACK_REF
|
#ifdef RADEON_BO_TRACK
|
||||||
_radeon_bo_debug(bo, 2, file, func, line);
|
radeon_track_add_event(bo->track, file, func, "ref", line);
|
||||||
#endif
|
#endif
|
||||||
bo->bom->funcs->bo_ref(bo);
|
bo->bom->funcs->bo_ref(bo);
|
||||||
}
|
}
|
||||||
|
|
||||||
static inline void _radeon_bo_unref(struct radeon_bo *bo,
|
static inline struct radeon_bo *_radeon_bo_unref(struct radeon_bo *bo,
|
||||||
const char *file,
|
const char *file,
|
||||||
const char *func,
|
const char *func,
|
||||||
int line)
|
int line)
|
||||||
{
|
{
|
||||||
bo->cref--;
|
bo->cref--;
|
||||||
#ifdef RADEON_BO_TRACK_REF
|
#ifdef RADEON_BO_TRACK
|
||||||
_radeon_bo_debug(bo, 3, file, func, line);
|
radeon_track_add_event(bo->track, file, func, "unref", line);
|
||||||
|
if (bo->cref <= 0) {
|
||||||
|
radeon_tracker_remove_track(&bo->bom->tracker, bo->track);
|
||||||
|
bo->track = NULL;
|
||||||
|
}
|
||||||
#endif
|
#endif
|
||||||
bo->bom->funcs->bo_unref(bo);
|
return bo->bom->funcs->bo_unref(bo);
|
||||||
}
|
}
|
||||||
|
|
||||||
static inline int _radeon_bo_map(struct radeon_bo *bo,
|
static inline int _radeon_bo_map(struct radeon_bo *bo,
|
||||||
|
|
@ -129,9 +140,6 @@ static inline int _radeon_bo_map(struct radeon_bo *bo,
|
||||||
const char *func,
|
const char *func,
|
||||||
int line)
|
int line)
|
||||||
{
|
{
|
||||||
#ifdef RADEON_BO_TRACK_MAP
|
|
||||||
_radeon_bo_debug(bo, 4, file, func, line);
|
|
||||||
#endif
|
|
||||||
return bo->bom->funcs->bo_map(bo, write);
|
return bo->bom->funcs->bo_map(bo, write);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
@ -140,9 +148,6 @@ static inline int _radeon_bo_unmap(struct radeon_bo *bo,
|
||||||
const char *func,
|
const char *func,
|
||||||
int line)
|
int line)
|
||||||
{
|
{
|
||||||
#ifdef RADEON_BO_TRACK_MAP
|
|
||||||
_radeon_bo_debug(bo, 5, file, func, line);
|
|
||||||
#endif
|
|
||||||
return bo->bom->funcs->bo_unmap(bo);
|
return bo->bom->funcs->bo_unmap(bo);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
||||||
|
|
@ -81,7 +81,6 @@ static struct radeon_bo *bo_open(struct radeon_bo_manager *bom,
|
||||||
open_arg.name = handle;
|
open_arg.name = handle;
|
||||||
r = ioctl(bom->fd, DRM_IOCTL_GEM_OPEN, &open_arg);
|
r = ioctl(bom->fd, DRM_IOCTL_GEM_OPEN, &open_arg);
|
||||||
if (r != 0) {
|
if (r != 0) {
|
||||||
fprintf(stderr, "GEM open failed: %d (%s)\n",r,strerror(r));
|
|
||||||
free(bo);
|
free(bo);
|
||||||
return NULL;
|
return NULL;
|
||||||
}
|
}
|
||||||
|
|
@ -95,6 +94,7 @@ static struct radeon_bo *bo_open(struct radeon_bo_manager *bom,
|
||||||
args.alignment = alignment;
|
args.alignment = alignment;
|
||||||
args.initial_domain = bo->base.domains;
|
args.initial_domain = bo->base.domains;
|
||||||
args.no_backing_store = 0;
|
args.no_backing_store = 0;
|
||||||
|
args.handle = 0;
|
||||||
r = drmCommandWriteRead(bom->fd, DRM_RADEON_GEM_CREATE,
|
r = drmCommandWriteRead(bom->fd, DRM_RADEON_GEM_CREATE,
|
||||||
&args, sizeof(args));
|
&args, sizeof(args));
|
||||||
bo->base.handle = args.handle;
|
bo->base.handle = args.handle;
|
||||||
|
|
@ -115,16 +115,16 @@ static void bo_ref(struct radeon_bo *bo)
|
||||||
{
|
{
|
||||||
}
|
}
|
||||||
|
|
||||||
static void bo_unref(struct radeon_bo *bo)
|
static struct radeon_bo *bo_unref(struct radeon_bo *bo)
|
||||||
{
|
{
|
||||||
struct radeon_bo_gem *bo_gem = (struct radeon_bo_gem*)bo;
|
struct radeon_bo_gem *bo_gem = (struct radeon_bo_gem*)bo;
|
||||||
struct drm_gem_close args;
|
struct drm_gem_close args;
|
||||||
|
|
||||||
if (bo == NULL) {
|
if (bo == NULL) {
|
||||||
return;
|
return NULL;
|
||||||
}
|
}
|
||||||
if (bo->cref) {
|
if (bo->cref) {
|
||||||
return;
|
return bo;
|
||||||
}
|
}
|
||||||
if (bo_gem->map_count) {
|
if (bo_gem->map_count) {
|
||||||
munmap(bo->ptr, bo->size);
|
munmap(bo->ptr, bo->size);
|
||||||
|
|
@ -133,7 +133,9 @@ static void bo_unref(struct radeon_bo *bo)
|
||||||
/* close object */
|
/* close object */
|
||||||
args.handle = bo->handle;
|
args.handle = bo->handle;
|
||||||
ioctl(bo->bom->fd, DRM_IOCTL_GEM_CLOSE, &args);
|
ioctl(bo->bom->fd, DRM_IOCTL_GEM_CLOSE, &args);
|
||||||
|
memset(bo_gem, 0, sizeof(struct radeon_bo_gem));
|
||||||
free(bo_gem);
|
free(bo_gem);
|
||||||
|
return NULL;
|
||||||
}
|
}
|
||||||
|
|
||||||
static int bo_map(struct radeon_bo *bo, int write)
|
static int bo_map(struct radeon_bo *bo, int write)
|
||||||
|
|
@ -182,7 +184,7 @@ static struct radeon_bo_funcs bo_gem_funcs = {
|
||||||
bo_unmap
|
bo_unmap
|
||||||
};
|
};
|
||||||
|
|
||||||
struct radeon_bo_manager *radeon_bo_manager_gem(int fd)
|
struct radeon_bo_manager *radeon_bo_manager_gem_ctor(int fd)
|
||||||
{
|
{
|
||||||
struct bo_manager_gem *bomg;
|
struct bo_manager_gem *bomg;
|
||||||
|
|
||||||
|
|
@ -195,7 +197,7 @@ struct radeon_bo_manager *radeon_bo_manager_gem(int fd)
|
||||||
return (struct radeon_bo_manager*)bomg;
|
return (struct radeon_bo_manager*)bomg;
|
||||||
}
|
}
|
||||||
|
|
||||||
void radeon_bo_manager_gem_shutdown(struct radeon_bo_manager *bom)
|
void radeon_bo_manager_gem_dtor(struct radeon_bo_manager *bom)
|
||||||
{
|
{
|
||||||
struct bo_manager_gem *bomg = (struct bo_manager_gem*)bom;
|
struct bo_manager_gem *bomg = (struct bo_manager_gem*)bom;
|
||||||
|
|
||||||
|
|
|
||||||
|
|
@ -34,7 +34,7 @@
|
||||||
|
|
||||||
#include "radeon_bo.h"
|
#include "radeon_bo.h"
|
||||||
|
|
||||||
struct radeon_bo_manager *radeon_bo_manager_gem(int fd);
|
struct radeon_bo_manager *radeon_bo_manager_gem_ctor(int fd);
|
||||||
void radeon_bo_manager_gem_shutdown(struct radeon_bo_manager *bom);
|
void radeon_bo_manager_gem_dtor(struct radeon_bo_manager *bom);
|
||||||
|
|
||||||
#endif
|
#endif
|
||||||
|
|
|
||||||
|
|
@ -87,6 +87,7 @@ struct radeon_cs_funcs {
|
||||||
int (*cs_destroy)(struct radeon_cs *cs);
|
int (*cs_destroy)(struct radeon_cs *cs);
|
||||||
int (*cs_erase)(struct radeon_cs *cs);
|
int (*cs_erase)(struct radeon_cs *cs);
|
||||||
int (*cs_need_flush)(struct radeon_cs *cs);
|
int (*cs_need_flush)(struct radeon_cs *cs);
|
||||||
|
void (*cs_print)(struct radeon_cs *cs, FILE *file);
|
||||||
};
|
};
|
||||||
|
|
||||||
struct radeon_cs_manager {
|
struct radeon_cs_manager {
|
||||||
|
|
@ -159,4 +160,9 @@ static inline int radeon_cs_need_flush(struct radeon_cs *cs)
|
||||||
return cs->csm->funcs->cs_need_flush(cs);
|
return cs->csm->funcs->cs_need_flush(cs);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static inline void radeon_cs_print(struct radeon_cs *cs, FILE *file)
|
||||||
|
{
|
||||||
|
cs->csm->funcs->cs_print(cs, file);
|
||||||
|
}
|
||||||
|
|
||||||
#endif
|
#endif
|
||||||
|
|
|
||||||
|
|
@ -251,8 +251,11 @@ static int cs_gem_emit(struct radeon_cs *cs)
|
||||||
{
|
{
|
||||||
struct cs_gem *csg = (struct cs_gem*)cs;
|
struct cs_gem *csg = (struct cs_gem*)cs;
|
||||||
uint64_t chunk_array[2];
|
uint64_t chunk_array[2];
|
||||||
|
unsigned i;
|
||||||
int r;
|
int r;
|
||||||
|
|
||||||
|
csg->chunks[0].length_dw = cs->cdw;
|
||||||
|
|
||||||
chunk_array[0] = (uint64_t)(intptr_t)&csg->chunks[0];
|
chunk_array[0] = (uint64_t)(intptr_t)&csg->chunks[0];
|
||||||
chunk_array[1] = (uint64_t)(intptr_t)&csg->chunks[1];
|
chunk_array[1] = (uint64_t)(intptr_t)&csg->chunks[1];
|
||||||
|
|
||||||
|
|
@ -261,10 +264,11 @@ static int cs_gem_emit(struct radeon_cs *cs)
|
||||||
|
|
||||||
r = drmCommandWriteRead(cs->csm->fd, DRM_RADEON_CS2,
|
r = drmCommandWriteRead(cs->csm->fd, DRM_RADEON_CS2,
|
||||||
&csg->cs, sizeof(struct drm_radeon_cs2));
|
&csg->cs, sizeof(struct drm_radeon_cs2));
|
||||||
if (r) {
|
for (i = 0; i < csg->base.crelocs; i++) {
|
||||||
return r;
|
radeon_bo_unref(csg->relocs_bo[i]);
|
||||||
|
csg->relocs_bo[i] = NULL;
|
||||||
}
|
}
|
||||||
return 0;
|
return r;
|
||||||
}
|
}
|
||||||
|
|
||||||
static int cs_gem_destroy(struct radeon_cs *cs)
|
static int cs_gem_destroy(struct radeon_cs *cs)
|
||||||
|
|
@ -281,7 +285,16 @@ static int cs_gem_destroy(struct radeon_cs *cs)
|
||||||
static int cs_gem_erase(struct radeon_cs *cs)
|
static int cs_gem_erase(struct radeon_cs *cs)
|
||||||
{
|
{
|
||||||
struct cs_gem *csg = (struct cs_gem*)cs;
|
struct cs_gem *csg = (struct cs_gem*)cs;
|
||||||
|
unsigned i;
|
||||||
|
|
||||||
|
if (csg->relocs_bo) {
|
||||||
|
for (i = 0; i < csg->base.crelocs; i++) {
|
||||||
|
if (csg->relocs_bo[i]) {
|
||||||
|
radeon_bo_unref(csg->relocs_bo[i]);
|
||||||
|
csg->relocs_bo[i] = NULL;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
cs->relocs_total_size = 0;
|
cs->relocs_total_size = 0;
|
||||||
cs->cdw = 0;
|
cs->cdw = 0;
|
||||||
cs->section = 0;
|
cs->section = 0;
|
||||||
|
|
@ -293,7 +306,103 @@ static int cs_gem_erase(struct radeon_cs *cs)
|
||||||
|
|
||||||
static int cs_gem_need_flush(struct radeon_cs *cs)
|
static int cs_gem_need_flush(struct radeon_cs *cs)
|
||||||
{
|
{
|
||||||
return (cs->relocs_total_size > (16*1024*1024));
|
return (cs->relocs_total_size > (32*1024*1024));
|
||||||
|
}
|
||||||
|
|
||||||
|
#define PACKET_TYPE0 0
|
||||||
|
#define PACKET_TYPE1 1
|
||||||
|
#define PACKET_TYPE2 2
|
||||||
|
#define PACKET_TYPE3 3
|
||||||
|
|
||||||
|
#define PACKET3_NOP 0x10
|
||||||
|
#define PACKET3_SET_SCISSORS 0x1E
|
||||||
|
#define PACKET3_3D_DRAW_VBUF 0x28
|
||||||
|
#define PACKET3_3D_DRAW_IMMD 0x29
|
||||||
|
#define PACKET3_3D_DRAW_INDX 0x2A
|
||||||
|
#define PACKET3_3D_LOAD_VBPNTR 0x2F
|
||||||
|
#define PACKET3_INDX_BUFFER 0x33
|
||||||
|
#define PACKET3_3D_DRAW_VBUF_2 0x34
|
||||||
|
#define PACKET3_3D_DRAW_IMMD_2 0x35
|
||||||
|
#define PACKET3_3D_DRAW_INDX_2 0x36
|
||||||
|
|
||||||
|
#define CP_PACKET_GET_TYPE(h) (((h) >> 30) & 3)
|
||||||
|
#define CP_PACKET_GET_COUNT(h) (((h) >> 16) & 0x3FFF)
|
||||||
|
#define CP_PACKET0_GET_REG(h) (((h) & 0x1FFF) << 2)
|
||||||
|
#define CP_PACKET0_GET_ONE_REG_WR(h) (((h) >> 15) & 1)
|
||||||
|
#define CP_PACKET3_GET_OPCODE(h) (((h) >> 8) & 0xFF)
|
||||||
|
|
||||||
|
static void cs_gem_print(struct radeon_cs *cs, FILE *file)
|
||||||
|
{
|
||||||
|
unsigned opcode;
|
||||||
|
unsigned reg;
|
||||||
|
unsigned cnt;
|
||||||
|
int i, j;
|
||||||
|
|
||||||
|
for (i = 0; i < cs->cdw;) {
|
||||||
|
cnt = CP_PACKET_GET_COUNT(cs->packets[i]);
|
||||||
|
switch (CP_PACKET_GET_TYPE(cs->packets[i])) {
|
||||||
|
case PACKET_TYPE0:
|
||||||
|
fprintf(file, "Pkt0 at %d (%d dwords):\n", i, cnt + 1);
|
||||||
|
reg = CP_PACKET0_GET_REG(cs->packets[i]);
|
||||||
|
if (CP_PACKET0_GET_ONE_REG_WR(cs->packets[i++])) {
|
||||||
|
for (j = 0; j <= cnt; j++) {
|
||||||
|
fprintf(file, " 0x%08X -> 0x%04X\n",
|
||||||
|
cs->packets[i++], reg);
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
for (j = 0; j <= cnt; j++) {
|
||||||
|
fprintf(file, " 0x%08X -> 0x%04X\n",
|
||||||
|
cs->packets[i++], reg);
|
||||||
|
reg += 4;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
break;
|
||||||
|
case PACKET_TYPE3:
|
||||||
|
fprintf(file, "Pkt3 at %d :\n", i);
|
||||||
|
opcode = CP_PACKET3_GET_OPCODE(cs->packets[i++]);
|
||||||
|
switch (opcode) {
|
||||||
|
case PACKET3_NOP:
|
||||||
|
fprintf(file, " PACKET3_NOP:\n");
|
||||||
|
break;
|
||||||
|
case PACKET3_3D_DRAW_VBUF:
|
||||||
|
fprintf(file, " PACKET3_3D_DRAW_VBUF:\n");
|
||||||
|
break;
|
||||||
|
case PACKET3_3D_DRAW_IMMD:
|
||||||
|
fprintf(file, " PACKET3_3D_DRAW_IMMD:\n");
|
||||||
|
break;
|
||||||
|
case PACKET3_3D_DRAW_INDX:
|
||||||
|
fprintf(file, " PACKET3_3D_DRAW_INDX:\n");
|
||||||
|
break;
|
||||||
|
case PACKET3_3D_LOAD_VBPNTR:
|
||||||
|
fprintf(file, " PACKET3_3D_LOAD_VBPNTR:\n");
|
||||||
|
break;
|
||||||
|
case PACKET3_INDX_BUFFER:
|
||||||
|
fprintf(file, " PACKET3_INDX_BUFFER:\n");
|
||||||
|
break;
|
||||||
|
case PACKET3_3D_DRAW_VBUF_2:
|
||||||
|
fprintf(file, " PACKET3_3D_DRAW_VBUF_2:\n");
|
||||||
|
break;
|
||||||
|
case PACKET3_3D_DRAW_IMMD_2:
|
||||||
|
fprintf(file, " PACKET3_3D_DRAW_IMMD_2:\n");
|
||||||
|
break;
|
||||||
|
case PACKET3_3D_DRAW_INDX_2:
|
||||||
|
fprintf(file, " PACKET3_3D_DRAW_INDX_2:\n");
|
||||||
|
break;
|
||||||
|
default:
|
||||||
|
fprintf(file, "Unknow opcode 0x%02X at %d\n", opcode, i);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
for (j = 0; j <= cnt; j++) {
|
||||||
|
fprintf(file, " 0x%08X\n", cs->packets[i++]);
|
||||||
|
}
|
||||||
|
break;
|
||||||
|
case PACKET_TYPE1:
|
||||||
|
case PACKET_TYPE2:
|
||||||
|
default:
|
||||||
|
fprintf(file, "Unknow packet 0x%08X at %d\n", cs->packets[i], i);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
static struct radeon_cs_funcs radeon_cs_gem_funcs = {
|
static struct radeon_cs_funcs radeon_cs_gem_funcs = {
|
||||||
|
|
@ -305,10 +414,11 @@ static struct radeon_cs_funcs radeon_cs_gem_funcs = {
|
||||||
cs_gem_emit,
|
cs_gem_emit,
|
||||||
cs_gem_destroy,
|
cs_gem_destroy,
|
||||||
cs_gem_erase,
|
cs_gem_erase,
|
||||||
cs_gem_need_flush
|
cs_gem_need_flush,
|
||||||
|
cs_gem_print
|
||||||
};
|
};
|
||||||
|
|
||||||
struct radeon_cs_manager *radeon_cs_manager_gem(int fd)
|
struct radeon_cs_manager *radeon_cs_manager_gem_ctor(int fd)
|
||||||
{
|
{
|
||||||
struct radeon_cs_manager *csm;
|
struct radeon_cs_manager *csm;
|
||||||
|
|
||||||
|
|
@ -322,7 +432,7 @@ struct radeon_cs_manager *radeon_cs_manager_gem(int fd)
|
||||||
return csm;
|
return csm;
|
||||||
}
|
}
|
||||||
|
|
||||||
void radeon_cs_manager_gem_shutdown(struct radeon_cs_manager *csm)
|
void radeon_cs_manager_gem_dtor(struct radeon_cs_manager *csm)
|
||||||
{
|
{
|
||||||
free(csm);
|
free(csm);
|
||||||
}
|
}
|
||||||
|
|
|
||||||
|
|
@ -35,7 +35,7 @@
|
||||||
|
|
||||||
#include "radeon_cs.h"
|
#include "radeon_cs.h"
|
||||||
|
|
||||||
struct radeon_cs_manager *radeon_cs_manager_gem(int fd);
|
struct radeon_cs_manager *radeon_cs_manager_gem_ctor(int fd);
|
||||||
void radeon_cs_manager_gem_shutdown(struct radeon_cs_manager *csm);
|
void radeon_cs_manager_gem_dtor(struct radeon_cs_manager *csm);
|
||||||
|
|
||||||
#endif
|
#endif
|
||||||
|
|
|
||||||
140
libdrm/radeon/radeon_track.c
Normal file
140
libdrm/radeon/radeon_track.c
Normal file
|
|
@ -0,0 +1,140 @@
|
||||||
|
/*
|
||||||
|
* Copyright © 2008 Jérôme Glisse
|
||||||
|
* All Rights Reserved.
|
||||||
|
*
|
||||||
|
* 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, sub license, 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 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
|
||||||
|
* NON-INFRINGEMENT. IN NO EVENT SHALL THE COPYRIGHT HOLDERS, AUTHORS
|
||||||
|
* AND/OR ITS SUPPLIERS 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.
|
||||||
|
*
|
||||||
|
* The above copyright notice and this permission notice (including the
|
||||||
|
* next paragraph) shall be included in all copies or substantial portions
|
||||||
|
* of the Software.
|
||||||
|
*/
|
||||||
|
/*
|
||||||
|
* Authors:
|
||||||
|
* Jérôme Glisse <glisse@freedesktop.org>
|
||||||
|
*/
|
||||||
|
#include <stdio.h>
|
||||||
|
#include <stdlib.h>
|
||||||
|
#include <string.h>
|
||||||
|
#include "radeon_track.h"
|
||||||
|
|
||||||
|
void radeon_track_add_event(struct radeon_track *track,
|
||||||
|
const char *file,
|
||||||
|
const char *func,
|
||||||
|
const char *op,
|
||||||
|
unsigned line)
|
||||||
|
{
|
||||||
|
struct radeon_track_event *event;
|
||||||
|
|
||||||
|
if (track == NULL) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
event = (void*)calloc(1,sizeof(struct radeon_track_event));
|
||||||
|
if (event == NULL) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
event->line = line;
|
||||||
|
event->file = strdup(file);
|
||||||
|
event->func = strdup(func);
|
||||||
|
event->op = strdup(op);
|
||||||
|
if (event->file == NULL || event->func == NULL || event->op == NULL) {
|
||||||
|
free(event->file);
|
||||||
|
free(event->func);
|
||||||
|
free(event->op);
|
||||||
|
free(event);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
event->next = track->events;
|
||||||
|
track->events = event;
|
||||||
|
}
|
||||||
|
|
||||||
|
struct radeon_track *radeon_tracker_add_track(struct radeon_tracker *tracker,
|
||||||
|
unsigned key)
|
||||||
|
{
|
||||||
|
struct radeon_track *track;
|
||||||
|
|
||||||
|
track = (struct radeon_track*)calloc(1, sizeof(struct radeon_track));
|
||||||
|
if (track) {
|
||||||
|
track->next = tracker->tracks.next;
|
||||||
|
track->prev = &tracker->tracks;
|
||||||
|
tracker->tracks.next = track;
|
||||||
|
if (track->next) {
|
||||||
|
track->next->prev = track;
|
||||||
|
}
|
||||||
|
track->key = key;
|
||||||
|
track->events = NULL;
|
||||||
|
}
|
||||||
|
return track;
|
||||||
|
}
|
||||||
|
|
||||||
|
void radeon_tracker_remove_track(struct radeon_tracker *tracker,
|
||||||
|
struct radeon_track *track)
|
||||||
|
{
|
||||||
|
struct radeon_track_event *event;
|
||||||
|
void *tmp;
|
||||||
|
|
||||||
|
if (track == NULL) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
track->prev->next = track->next;
|
||||||
|
if (track->next) {
|
||||||
|
track->next->prev = track->prev;
|
||||||
|
}
|
||||||
|
track->next = track->prev = NULL;
|
||||||
|
event = track->events;
|
||||||
|
while (event) {
|
||||||
|
tmp = event;
|
||||||
|
free(event->file);
|
||||||
|
free(event->func);
|
||||||
|
free(event->op);
|
||||||
|
event = event->next;
|
||||||
|
free(tmp);
|
||||||
|
}
|
||||||
|
track->events = NULL;
|
||||||
|
free(track);
|
||||||
|
}
|
||||||
|
|
||||||
|
void radeon_tracker_print(struct radeon_tracker *tracker, FILE *file)
|
||||||
|
{
|
||||||
|
struct radeon_track *track;
|
||||||
|
struct radeon_track_event *event;
|
||||||
|
void *tmp;
|
||||||
|
|
||||||
|
track = tracker->tracks.next;
|
||||||
|
while (track) {
|
||||||
|
event = track->events;
|
||||||
|
fprintf(file, "[0x%08X] :\n", track->key);
|
||||||
|
while (event) {
|
||||||
|
tmp = event;
|
||||||
|
fprintf(file, " [0x%08X:%s](%s:%s:%d)\n",
|
||||||
|
track->key, event->op, event->file,
|
||||||
|
event->func, event->line);
|
||||||
|
free(event->file);
|
||||||
|
free(event->func);
|
||||||
|
free(event->op);
|
||||||
|
event->file = NULL;
|
||||||
|
event->func = NULL;
|
||||||
|
event->op = NULL;
|
||||||
|
event = event->next;
|
||||||
|
free(tmp);
|
||||||
|
}
|
||||||
|
track->events = NULL;
|
||||||
|
tmp = track;
|
||||||
|
track = track->next;
|
||||||
|
free(tmp);
|
||||||
|
}
|
||||||
|
}
|
||||||
64
libdrm/radeon/radeon_track.h
Normal file
64
libdrm/radeon/radeon_track.h
Normal file
|
|
@ -0,0 +1,64 @@
|
||||||
|
/*
|
||||||
|
* Copyright © 2008 Jérôme Glisse
|
||||||
|
* All Rights Reserved.
|
||||||
|
*
|
||||||
|
* 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, sub license, 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 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
|
||||||
|
* NON-INFRINGEMENT. IN NO EVENT SHALL THE COPYRIGHT HOLDERS, AUTHORS
|
||||||
|
* AND/OR ITS SUPPLIERS 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.
|
||||||
|
*
|
||||||
|
* The above copyright notice and this permission notice (including the
|
||||||
|
* next paragraph) shall be included in all copies or substantial portions
|
||||||
|
* of the Software.
|
||||||
|
*/
|
||||||
|
/*
|
||||||
|
* Authors:
|
||||||
|
* Jérôme Glisse <glisse@freedesktop.org>
|
||||||
|
*/
|
||||||
|
#ifndef RADEON_TRACK_H
|
||||||
|
#define RADEON_TRACK_H
|
||||||
|
|
||||||
|
struct radeon_track_event {
|
||||||
|
struct radeon_track_event *next;
|
||||||
|
char *file;
|
||||||
|
char *func;
|
||||||
|
char *op;
|
||||||
|
unsigned line;
|
||||||
|
};
|
||||||
|
|
||||||
|
struct radeon_track {
|
||||||
|
struct radeon_track *next;
|
||||||
|
struct radeon_track *prev;
|
||||||
|
unsigned key;
|
||||||
|
struct radeon_track_event *events;
|
||||||
|
};
|
||||||
|
|
||||||
|
struct radeon_tracker {
|
||||||
|
struct radeon_track tracks;
|
||||||
|
};
|
||||||
|
|
||||||
|
void radeon_track_add_event(struct radeon_track *track,
|
||||||
|
const char *file,
|
||||||
|
const char *func,
|
||||||
|
const char *op,
|
||||||
|
unsigned line);
|
||||||
|
struct radeon_track *radeon_tracker_add_track(struct radeon_tracker *tracker,
|
||||||
|
unsigned key);
|
||||||
|
void radeon_tracker_remove_track(struct radeon_tracker *tracker,
|
||||||
|
struct radeon_track *track);
|
||||||
|
void radeon_tracker_print(struct radeon_tracker *tracker,
|
||||||
|
FILE *file);
|
||||||
|
|
||||||
|
#endif
|
||||||
|
|
@ -785,8 +785,8 @@ out:
|
||||||
}
|
}
|
||||||
drm_bo_add_to_lru(bo);
|
drm_bo_add_to_lru(bo);
|
||||||
BUG_ON(bo->priv_flags & _DRM_BO_FLAG_UNLOCKED);
|
BUG_ON(bo->priv_flags & _DRM_BO_FLAG_UNLOCKED);
|
||||||
out_unlock:
|
|
||||||
mutex_unlock(&dev->struct_mutex);
|
mutex_unlock(&dev->struct_mutex);
|
||||||
|
out_unlock:
|
||||||
|
|
||||||
return ret;
|
return ret;
|
||||||
}
|
}
|
||||||
|
|
|
||||||
|
|
@ -41,21 +41,25 @@ int radeon_cs2_ioctl(struct drm_device *dev, void *data, struct drm_file *fpriv)
|
||||||
long size;
|
long size;
|
||||||
int r, i;
|
int r, i;
|
||||||
|
|
||||||
|
mutex_lock(&dev_priv->cs.cs_mutex);
|
||||||
/* set command stream id to 0 which is fake id */
|
/* set command stream id to 0 which is fake id */
|
||||||
cs_id = 0;
|
cs_id = 0;
|
||||||
cs->cs_id = cs_id;
|
cs->cs_id = cs_id;
|
||||||
|
|
||||||
if (dev_priv == NULL) {
|
if (dev_priv == NULL) {
|
||||||
DRM_ERROR("called with no initialization\n");
|
DRM_ERROR("called with no initialization\n");
|
||||||
|
mutex_unlock(&dev_priv->cs.cs_mutex);
|
||||||
return -EINVAL;
|
return -EINVAL;
|
||||||
}
|
}
|
||||||
if (!cs->num_chunks) {
|
if (!cs->num_chunks) {
|
||||||
|
mutex_unlock(&dev_priv->cs.cs_mutex);
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
chunk_array = drm_calloc(cs->num_chunks, sizeof(uint64_t), DRM_MEM_DRIVER);
|
chunk_array = drm_calloc(cs->num_chunks, sizeof(uint64_t), DRM_MEM_DRIVER);
|
||||||
if (!chunk_array) {
|
if (!chunk_array) {
|
||||||
|
mutex_unlock(&dev_priv->cs.cs_mutex);
|
||||||
return -ENOMEM;
|
return -ENOMEM;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
@ -161,6 +165,7 @@ int radeon_cs2_ioctl(struct drm_device *dev, void *data, struct drm_file *fpriv)
|
||||||
|
|
||||||
out:
|
out:
|
||||||
dev_priv->cs.ib_free(&parser);
|
dev_priv->cs.ib_free(&parser);
|
||||||
|
mutex_unlock(&dev_priv->cs.cs_mutex);
|
||||||
|
|
||||||
for (i = 0; i < parser.num_chunks; i++) {
|
for (i = 0; i < parser.num_chunks; i++) {
|
||||||
if (parser.chunks[i].kdata)
|
if (parser.chunks[i].kdata)
|
||||||
|
|
@ -646,6 +651,7 @@ int radeon_cs_init(struct drm_device *dev)
|
||||||
{
|
{
|
||||||
drm_radeon_private_t *dev_priv = dev->dev_private;
|
drm_radeon_private_t *dev_priv = dev->dev_private;
|
||||||
|
|
||||||
|
mutex_init(&dev_priv->cs.cs_mutex);
|
||||||
if (dev_priv->chip_family < CHIP_RV280) {
|
if (dev_priv->chip_family < CHIP_RV280) {
|
||||||
dev_priv->cs.id_emit = r100_cs_id_emit;
|
dev_priv->cs.id_emit = r100_cs_id_emit;
|
||||||
dev_priv->cs.id_last_get = r100_cs_id_last_get;
|
dev_priv->cs.id_last_get = r100_cs_id_last_get;
|
||||||
|
|
|
||||||
|
|
@ -316,6 +316,7 @@ struct drm_radeon_cs_parser {
|
||||||
|
|
||||||
/* command submission struct */
|
/* command submission struct */
|
||||||
struct drm_radeon_cs_priv {
|
struct drm_radeon_cs_priv {
|
||||||
|
struct mutex cs_mutex;
|
||||||
uint32_t id_wcnt;
|
uint32_t id_wcnt;
|
||||||
uint32_t id_scnt;
|
uint32_t id_scnt;
|
||||||
uint32_t id_last_wcnt;
|
uint32_t id_last_wcnt;
|
||||||
|
|
|
||||||
Loading…
Add table
Reference in a new issue