c51cbc6c9f6124abd78ea9e650f0bc272375cefa
[platform/upstream/libdrm.git] / tests / drmstat.c
1 /* drmstat.c -- DRM device status and testing program
2  * Created: Tue Jan  5 08:19:24 1999 by faith@precisioninsight.com
3  *
4  * Copyright 1999 Precision Insight, Inc., Cedar Park, Texas.
5  * Copyright 2000 VA Linux Systems, Inc., Sunnyvale, California.
6  * All Rights Reserved.
7  *
8  * Permission is hereby granted, free of charge, to any person obtaining a
9  * copy of this software and associated documentation files (the "Software"),
10  * to deal in the Software without restriction, including without limitation
11  * the rights to use, copy, modify, merge, publish, distribute, sublicense,
12  * and/or sell copies of the Software, and to permit persons to whom the
13  * Software is furnished to do so, subject to the following conditions:
14  * 
15  * The above copyright notice and this permission notice (including the next
16  * paragraph) shall be included in all copies or substantial portions of the
17  * Software.
18  * 
19  * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
20  * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
21  * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT.  IN NO EVENT SHALL
22  * PRECISION INSIGHT AND/OR ITS SUPPLIERS BE LIABLE FOR ANY CLAIM, DAMAGES OR
23  * OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE,
24  * ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER
25  * DEALINGS IN THE SOFTWARE.
26  * 
27  * Authors: Rickard E. (Rik) Faith <faith@valinux.com>
28  * 
29  */
30
31 #include "config.h"
32
33 #include <stdio.h>
34 #include <stdlib.h>
35 #include <unistd.h>
36 #include <sys/types.h>
37 #include <sys/time.h>
38 #include <sys/mman.h>
39 #include <getopt.h>
40 #include <strings.h>
41 #include <errno.h>
42 #include <signal.h>
43 #include <fcntl.h>
44 #ifdef HAVE_ALLOCA_H
45 # include <alloca.h>
46 #endif
47 #include "xf86drm.h"
48
49 /* Support gcc's __FUNCTION__ for people using other compilers */
50 #if !defined(__GNUC__) && !defined(__FUNCTION__)
51 # define __FUNCTION__ __func__ /* C99 */
52 #endif
53
54 int sigio_fd;
55
56 static double usec(struct timeval *end, struct timeval *start)
57 {
58     double e = end->tv_sec   * 1000000 + end->tv_usec;
59     double s = start->tv_sec * 1000000 + start->tv_usec;
60
61     return e - s;
62 }
63
64 static void getversion(int fd)
65 {
66     drmVersionPtr version;
67     
68     version = drmGetVersion(fd);
69     if (version) {
70         printf( "Name: %s\n", version->name ? version->name : "?" );
71         printf( "    Version: %d.%d.%d\n",
72                 version->version_major,
73                 version->version_minor,
74                 version->version_patchlevel );
75         printf( "    Date: %s\n", version->date ? version->date : "?" );
76         printf( "    Desc: %s\n", version->desc ? version->desc : "?" );
77         drmFreeVersion(version);
78     } else {
79         printf( "No driver available\n" );
80     }
81 }
82         
83 void handler(int fd, void *oldctx, void *newctx)
84 {
85     printf("Got fd %d\n", fd);
86 }
87
88 void process_sigio(char *device)
89 {
90     int              fd;
91
92     if ((fd = open(device, 0)) < 0) {
93         drmError(-errno, __FUNCTION__);
94         exit(1);
95     }
96
97     sigio_fd = fd;
98     /*  drmInstallSIGIOHandler(fd, handler); */
99     for (;;) sleep(60);
100 }
101
102 int main(int argc, char **argv)
103 {
104     int            c;
105     int            r  = 0;
106     int            fd = -1;
107     drm_handle_t      handle;
108     void           *address;
109     char           *pt;
110     unsigned long  count;
111     unsigned long  offset;
112     unsigned long  size;
113     drm_context_t  context;
114     int            loops;
115     char           buf[1024];
116     int            i;
117     drmBufInfoPtr  info;
118     drmBufMapPtr   bufs;
119     drmLockPtr     lock;
120     int            secs;
121
122     while ((c = getopt(argc, argv,
123                        "lc:vo:O:f:s:w:W:b:r:R:P:L:C:XS:B:F:")) != EOF)
124         switch (c) {
125         case 'F':
126             count  = strtoul(optarg, NULL, 0);
127             if (!fork()) {
128                 dup(fd);
129                 sleep(count);
130             }
131             close(fd);
132             break;
133         case 'v': getversion(fd);                                        break;
134         case 'X':
135             if ((r = drmCreateContext(fd, &context))) {
136                 drmError(r, argv[0]);
137                 return 1;
138             }
139             printf( "Got %d\n", context);
140             break;
141         case 'S':
142             process_sigio(optarg);
143             break;
144         case 'C':
145             if ((r = drmSwitchToContext(fd, strtoul(optarg, NULL, 0)))) {
146                 drmError(r, argv[0]);
147                 return 1;
148             }
149             break;
150         case 'c':
151             if ((r = drmSetBusid(fd,optarg))) {
152                 drmError(r, argv[0]);
153                 return 1;
154             }
155             break;
156         case 'o':
157             if ((fd = drmOpen(optarg, NULL)) < 0) {
158                 drmError(fd, argv[0]);
159                 return 1;
160             }
161             break;
162         case 'O':
163             if ((fd = drmOpen(NULL, optarg)) < 0) {
164                 drmError(fd, argv[0]);
165                 return 1;
166             }
167             break;
168         case 'B':               /* Test buffer allocation */
169             count  = strtoul(optarg, &pt, 0);
170             size   = strtoul(pt+1, &pt, 0);
171             secs   = strtoul(pt+1, NULL, 0);
172             {
173                 drmDMAReq      dma;
174                 int            *indices, *sizes;
175
176                 indices = alloca(sizeof(*indices) * count);
177                 sizes   = alloca(sizeof(*sizes)   * count);
178                 dma.context         = context;
179                 dma.send_count      = 0;
180                 dma.request_count   = count;
181                 dma.request_size    = size;
182                 dma.request_list    = indices;
183                 dma.request_sizes   = sizes;
184                 dma.flags           = DRM_DMA_WAIT;
185                 if ((r = drmDMA(fd, &dma))) {
186                     drmError(r, argv[0]);
187                     return 1;
188                 }
189                 for (i = 0; i < dma.granted_count; i++) {
190                     printf("%5d: index = %d, size = %d\n",
191                            i, dma.request_list[i], dma.request_sizes[i]);
192                 }
193                 sleep(secs);
194                 drmFreeBufs(fd, dma.granted_count, indices);
195             }
196             break;
197         case 'b':
198             count   = strtoul(optarg, &pt, 0);
199             size    = strtoul(pt+1, NULL, 0);
200             if ((r = drmAddBufs(fd, count, size, 0, 65536)) < 0) {
201                 drmError(r, argv[0]);
202                 return 1;
203             }
204             if (!(info = drmGetBufInfo(fd))) {
205                 drmError(0, argv[0]);
206                 return 1;
207             }
208             for (i = 0; i < info->count; i++) {
209                 printf("%5d buffers of size %6d (low = %d, high = %d)\n",
210                        info->list[i].count,
211                        info->list[i].size,
212                        info->list[i].low_mark,
213                        info->list[i].high_mark);
214             }
215             if ((r = drmMarkBufs(fd, 0.50, 0.80))) {
216                 drmError(r, argv[0]);
217                 return 1;
218             }
219             if (!(info = drmGetBufInfo(fd))) {
220                 drmError(0, argv[0]);
221                 return 1;
222             }
223             for (i = 0; i < info->count; i++) {
224                 printf("%5d buffers of size %6d (low = %d, high = %d)\n",
225                        info->list[i].count,
226                        info->list[i].size,
227                        info->list[i].low_mark,
228                        info->list[i].high_mark);
229             }
230             printf("===== /proc/dri/0/mem =====\n");
231             sprintf(buf, "cat /proc/dri/0/mem");
232             system(buf);
233 #if 1
234             if (!(bufs = drmMapBufs(fd))) {
235                 drmError(0, argv[0]);
236                 return 1;
237             }
238             printf("===============================\n");
239             printf( "%d bufs\n", bufs->count);
240             for (i = 0; i < bufs->count; i++) {
241                 printf( "  %4d: %8d bytes at %p\n",
242                         i,
243                         bufs->list[i].total,
244                         bufs->list[i].address);
245             }
246             printf("===== /proc/dri/0/vma =====\n");
247             sprintf(buf, "cat /proc/dri/0/vma");
248             system(buf);
249 #endif
250             break;
251         case 'f':
252             offset  = strtoul(optarg, &pt, 0);
253             size    = strtoul(pt+1, NULL, 0);
254             handle  = 0;
255             if ((r = drmAddMap(fd, offset, size,
256                                DRM_FRAME_BUFFER, 0, &handle))) {
257                 drmError(r, argv[0]);
258                 return 1;
259             }
260             printf("0x%08lx:0x%04lx added\n", offset, size);
261             printf("===== /proc/dri/0/mem =====\n");
262             sprintf(buf, "cat /proc/dri/0/mem");
263             system(buf);
264             break;
265         case 'r':
266         case 'R':
267             offset  = strtoul(optarg, &pt, 0);
268             size    = strtoul(pt+1, NULL, 0);
269             handle  = 0;
270             if ((r = drmAddMap(fd, offset, size,
271                                DRM_REGISTERS,
272                                c == 'R' ? DRM_READ_ONLY : 0,
273                                &handle))) {
274                 drmError(r, argv[0]);
275                 return 1;
276             }
277             printf("0x%08lx:0x%04lx added\n", offset, size);
278             printf("===== /proc/dri/0/mem =====\n");
279             sprintf(buf, "cat /proc/dri/0/mem");
280             system(buf);
281             break;
282         case 's':
283             size = strtoul(optarg, &pt, 0);
284             handle = 0;
285             if ((r = drmAddMap(fd, 0, size,
286                                DRM_SHM, DRM_CONTAINS_LOCK,
287                                &handle))) {
288                 drmError(r, argv[0]);
289                 return 1;
290             }
291             printf("0x%04lx byte shm added at 0x%08lx\n", size, handle);
292             sprintf(buf, "cat /proc/dri/0/vm");
293             system(buf);
294             break;
295         case 'P':
296             offset  = strtoul(optarg, &pt, 0);
297             size    = strtoul(pt+1, NULL, 0);
298             address = NULL;
299             if ((r = drmMap(fd, offset, size, &address))) {
300                 drmError(r, argv[0]);
301                 return 1;
302             }
303             printf("0x%08lx:0x%04lx mapped at %p for pid %d\n",
304                    offset, size, address, getpid());
305             printf("===== /proc/dri/0/vma =====\n");
306             sprintf(buf, "cat /proc/dri/0/vma");
307             system(buf);
308             mprotect((void *)offset, size, PROT_READ);
309             printf("===== /proc/dri/0/vma =====\n");
310             sprintf(buf, "cat /proc/dri/0/vma");
311             system(buf);
312             break;
313         case 'w':
314         case 'W':
315             offset  = strtoul(optarg, &pt, 0);
316             size    = strtoul(pt+1, NULL, 0);
317             address = NULL;
318             if ((r = drmMap(fd, offset, size, &address))) {
319                 drmError(r, argv[0]);
320                 return 1;
321             }
322             printf("0x%08lx:0x%04lx mapped at %p for pid %d\n",
323                    offset, size, address, getpid());
324             printf("===== /proc/%d/maps =====\n", getpid());
325             sprintf(buf, "cat /proc/%d/maps", getpid());
326             system(buf);
327             printf("===== /proc/dri/0/mem =====\n");
328             sprintf(buf, "cat /proc/dri/0/mem");
329             system(buf);
330             printf("===== /proc/dri/0/vma =====\n");
331             sprintf(buf, "cat /proc/dri/0/vma");
332             system(buf);
333             printf("===== READING =====\n");
334             for (i = 0; i < 0x10; i++)
335                 printf("%02x ", (unsigned int)((unsigned char *)address)[i]);
336             printf("\n");
337             if (c == 'w') {
338                 printf("===== WRITING =====\n");
339                 for (i = 0; i < size; i+=2) {
340                     ((char *)address)[i]   = i & 0xff;
341                     ((char *)address)[i+1] = i & 0xff;
342                 }
343             }
344             printf("===== READING =====\n");
345             for (i = 0; i < 0x10; i++)
346                 printf("%02x ", (unsigned int)((unsigned char *)address)[i]);
347             printf("\n");
348             printf("===== /proc/dri/0/vma =====\n");
349             sprintf(buf, "cat /proc/dri/0/vma");
350             system(buf);
351             break;
352         case 'L':
353             context = strtoul(optarg, &pt, 0);
354             offset  = strtoul(pt+1, &pt, 0);
355             size    = strtoul(pt+1, &pt, 0);
356             loops   = strtoul(pt+1, NULL, 0);
357             address = NULL;
358             if ((r = drmMap(fd, offset, size, &address))) {
359                 drmError(r, argv[0]);
360                 return 1;
361             }
362             lock       = address;
363 #if 1
364             {
365                 int            counter = 0;
366                 struct timeval loop_start, loop_end;
367                 struct timeval lock_start, lock_end;
368                 double         wt;
369 #define HISTOSIZE 9
370                 int            histo[HISTOSIZE];
371                 int            output = 0;
372                 int            fast   = 0;
373
374                 if (loops < 0) {
375                     loops = -loops;
376                     ++output;
377                 }
378
379                 for (i = 0; i < HISTOSIZE; i++) histo[i] = 0;
380
381                 gettimeofday(&loop_start, NULL);
382                 for (i = 0; i < loops; i++) {
383                     gettimeofday(&lock_start, NULL);
384                     DRM_LIGHT_LOCK_COUNT(fd,lock,context,fast);
385                     gettimeofday(&lock_end, NULL);
386                     DRM_UNLOCK(fd,lock,context);
387                     ++counter;
388                     wt = usec(&lock_end, &lock_start);
389                     if      (wt <=      2.5) ++histo[8];
390                     if      (wt <       5.0) ++histo[0];
391                     else if (wt <      50.0) ++histo[1];
392                     else if (wt <     500.0) ++histo[2];
393                     else if (wt <    5000.0) ++histo[3];
394                     else if (wt <   50000.0) ++histo[4];
395                     else if (wt <  500000.0) ++histo[5];
396                     else if (wt < 5000000.0) ++histo[6];
397                     else                     ++histo[7];
398                     if (output) printf( "%.2f uSec, %d fast\n", wt, fast);
399                 }
400                 gettimeofday(&loop_end, NULL);
401                 printf( "Average wait time = %.2f usec, %d fast\n",
402                         usec(&loop_end, &loop_start) /  counter, fast);
403                 printf( "%9d <=     2.5 uS\n", histo[8]);
404                 printf( "%9d <        5 uS\n", histo[0]);
405                 printf( "%9d <       50 uS\n", histo[1]);
406                 printf( "%9d <      500 uS\n", histo[2]);
407                 printf( "%9d <     5000 uS\n", histo[3]);
408                 printf( "%9d <    50000 uS\n", histo[4]);
409                 printf( "%9d <   500000 uS\n", histo[5]);
410                 printf( "%9d <  5000000 uS\n", histo[6]);
411                 printf( "%9d >= 5000000 uS\n", histo[7]);
412             }
413 #else
414             printf( "before lock: 0x%08x\n", lock->lock);
415             printf( "lock: 0x%08x\n", lock->lock);
416             sleep(5);
417             printf( "unlock: 0x%08x\n", lock->lock);
418 #endif
419             break;
420         default:
421             fprintf( stderr, "Usage: drmstat [options]\n" );
422             return 1;
423         }
424
425     return r; 
426 }
427
428 void DRM_PRINTFLIKE(4, 0)
429 xf86VDrvMsgVerb(int scrnIndex, int type, int verb, const char *format,
430                 va_list args)
431 {
432         vfprintf(stderr, format, args);
433 }
434
435 int xf86ConfigDRI[10];