Mercurial > hg > minimon
changeset 2:bac8ed8d6eb9
add jpeg coding and test stuff
author | Peter Meerwald <pmeerw@pmeerw.net> |
---|---|
date | Sun, 08 May 2011 18:22:22 +0200 (2011-05-08) |
parents | 111d4bbce605 |
children | e8957085fe8a |
files | Makefile fb.c fbt.c frame-ctrl.py jpg.c minimon.c plug-mass_log.txt to-mass_log.txt to-mini_log.txt vfb/Makefile vfb/vfb.c |
diffstat | 10 files changed, 990 insertions(+), 16 deletions(-) [+] |
line wrap: on
line diff
--- a/Makefile Sat May 07 19:00:43 2011 +0200 +++ b/Makefile Sun May 08 18:22:22 2011 +0200 @@ -1,7 +1,10 @@ -all: minimon +all: minimon fbt -minimon: minimon.c - $(CC) $(CFLAGS) -Wall -g -o $@ minimon.c -lusb +minimon: minimon.c jpg.c + $(CC) $(CFLAGS) -Wall -g -o $@ minimon.c jpg.c fb.c -lusb -ljpeg + +fbt: fbt.c + $(CC) $(CFLAGS) -Wall -g -o $@ fbt.c clean: rm -f minimon
--- /dev/null Thu Jan 01 00:00:00 1970 +0000 +++ b/fbt.c Sun May 08 18:22:22 2011 +0200 @@ -0,0 +1,38 @@ +#include <stdlib.h> +#include <stdio.h> +#include <fcntl.h> +#include <sys/ioctl.h> + +#include "linux/fb.h" + +int main(int argc, char *argv[]) { + + int fd = open("/dev/fb0", O_RDWR); + if (fd < 0) { + perror(""); + exit(EXIT_FAILURE); + } + + struct fb_fix_screeninfo sif; + if (ioctl(fd, FBIOGET_VSCREENINFO, &sif) < 0) { + perror(""); + exit(EXIT_FAILURE); + } + + printf("id %s\n", sif.id); + printf("type %d, aux %d\n", sif.type, sif.type_aux); + printf("visual %d\n", sif.visual); + printf("accel %d\n", sif.accel); + printf("line length %d\n", sif.line_length); + + struct fb_var_screeninfo siv; + if (ioctl(fd, FBIOGET_VSCREENINFO, &siv) < 0) { + perror(""); + exit(EXIT_FAILURE); + } + + printf("res x %d y %d\n", siv.xres, siv.yres); + printf("bpp %d\n", siv.bits_per_pixel); + + exit(EXIT_SUCCESS); +}
--- a/frame-ctrl.py Sat May 07 19:00:43 2011 +0200 +++ b/frame-ctrl.py Sun May 08 18:22:22 2011 +0200 @@ -8,7 +8,10 @@ from usb.util import * vendorId = 0x04e8 -models = {'SPF-87H': (0x2033, 0x2034), 'SPF-107H': (0x2027, 0x2028) } +models = {'SPF-87H': (0x2033, 0x2034), + 'SPF-107H': (0x2027, 0x2028), + 'SPF-107H': (0x2035, 0x2036), + } chunkSize = 0x4000 bufferSize = 0x20000
--- /dev/null Thu Jan 01 00:00:00 1970 +0000 +++ b/jpg.c Sun May 08 18:22:22 2011 +0200 @@ -0,0 +1,124 @@ +#include <stdlib.h> +#include <stdio.h> +#include <string.h> +#include "jpeglib.h" +#include "jerror.h" + +typedef struct { + struct jpeg_destination_mgr pub; /* public fields */ + + unsigned char ** outbuffer; /* target buffer */ + unsigned long * outsize; + unsigned char * newbuffer; /* newly allocated buffer */ + JOCTET * buffer; /* start of buffer */ + size_t bufsize; +} mem_dest_mgr; + +typedef mem_dest_mgr * mem_dest_mgr_ptr; + +void init_mem_destination (j_compress_ptr cinfo) { +} + +boolean empty_mem_output_buffer(j_compress_ptr cinfo) { + size_t nextsize; + JOCTET * nextbuffer; + mem_dest_mgr_ptr dest = (mem_dest_mgr_ptr) cinfo->dest; + + /* Try to allocate new buffer with double size */ + nextsize = dest->bufsize * 2; + nextbuffer = malloc(nextsize); + + if (nextbuffer == NULL) + ERREXIT1(cinfo, JERR_OUT_OF_MEMORY, 10); + + memcpy(nextbuffer, dest->buffer, dest->bufsize); + + if (dest->newbuffer != NULL) + free(dest->newbuffer); + + dest->newbuffer = nextbuffer; + + dest->pub.next_output_byte = nextbuffer + dest->bufsize; + dest->pub.free_in_buffer = dest->bufsize; + + dest->buffer = nextbuffer; + dest->bufsize = nextsize; + + return TRUE; +} + +void term_mem_destination(j_compress_ptr cinfo) { + mem_dest_mgr_ptr dest = (mem_dest_mgr_ptr) cinfo->dest; + + *dest->outbuffer = dest->buffer; + *dest->outsize = dest->bufsize - dest->pub.free_in_buffer; +} + +void jpeg_mem_dest(j_compress_ptr cinfo, unsigned char ** outbuffer, unsigned long * outsize) { + mem_dest_mgr_ptr dest; + + if (outbuffer == NULL || outsize == NULL) /* sanity check */ + ERREXIT(cinfo, JERR_BUFFER_SIZE); + + /* The destination object is made permanent so that multiple JPEG images + * can be written to the same buffer without re-executing jpeg_mem_dest. + */ + if (cinfo->dest == NULL) { /* first time for this JPEG object? */ + cinfo->dest = (struct jpeg_destination_mgr *) + (*cinfo->mem->alloc_small) ((j_common_ptr) cinfo, JPOOL_PERMANENT, + sizeof(mem_dest_mgr)); + } + + dest = (mem_dest_mgr_ptr) cinfo->dest; + dest->pub.init_destination = init_mem_destination; + dest->pub.empty_output_buffer = empty_mem_output_buffer; + dest->pub.term_destination = term_mem_destination; + dest->outbuffer = outbuffer; + dest->outsize = outsize; + dest->newbuffer = NULL; + + if (*outbuffer == NULL || *outsize == 0) { + /* Allocate initial buffer */ + dest->newbuffer = *outbuffer = malloc(65536); + if (dest->newbuffer == NULL) + ERREXIT1(cinfo, JERR_OUT_OF_MEMORY, 10); + *outsize = 65536; + } + + dest->pub.next_output_byte = dest->buffer = *outbuffer; + dest->pub.free_in_buffer = dest->bufsize = *outsize; +} + + +int build_jpg(JSAMPLE *image_buf, int width, int height) { + int scanline; + struct jpeg_compress_struct cinfo; + struct jpeg_error_mgr jerr; + + cinfo.err = jpeg_std_error(&jerr); + jpeg_create_compress(&cinfo); + + unsigned long out_size = 1<<20; + unsigned char *out_buf = NULL; + jpeg_mem_dest(&cinfo, &out_buf, &out_size); + + cinfo.image_width = width; + cinfo.image_height = height; + cinfo.input_components = 3; + cinfo.in_color_space = JCS_RGB; + jpeg_set_defaults(&cinfo); + + jpeg_start_compress(&cinfo, TRUE); + + for (scanline = 0; scanline < height; scanline++) { + JSAMPROW *row_ptr; + row_ptr = &image_buf[scanline * width]; + jpeg_write_scanlines(&cinfo, row_ptr, 1); + } + + jpeg_finish_compress(&cinfo); + + jpeg_destroy_compress(&cinfo); + + return 0; +}
--- a/minimon.c Sat May 07 19:00:43 2011 +0200 +++ b/minimon.c Sun May 08 18:22:22 2011 +0200 @@ -18,10 +18,11 @@ int height; } id_info_t; +// there are many more, static id_info_t ids[] = { - {0x2027, 0x2028, }, + {0x2027, 0x2028, "SPF-107H", 1024, 600}, {0xffff, 0xffff, "SPF-75H", 800, 480}, - {0xffff, 0xffff, "SPF-83H", 800, 600}, // 85P?? + {0xffff, 0xffff, "SPF-83H", 800, 600}, {0xffff, 0xffff, "SPF-85H", 800, 600}, // 85P?? {0x2033, 0x2034, "SPF-87H", 800, 480}, {0x2035, 0x2036, "SPF-107H", 1024, 600}, @@ -76,7 +77,6 @@ static usb_dev_handle *dev_open(struct usb_device *dev) { int res = -1; - char buf[256]; usb_dev_handle *udev; int numeps = 0; @@ -86,7 +86,7 @@ exit(EXIT_FAILURE); } - setuid(getuid()); +// setuid(getuid()); res = usb_set_configuration(udev, 1); @@ -97,10 +97,6 @@ exit(EXIT_FAILURE); } - strcpy(buf, "** no string **"); - res = usb_get_string_simple(udev, dev->descriptor.iManufacturer, buf, sizeof(buf)); - // fprintf(stderr, "usb_get_string_simple => %d, %s\n", res, buf); - { int eplist[] = { 0x2, 0x81, 0x83 }; int eplength = sizeof(eplist)/sizeof(eplist[0]); @@ -179,6 +175,7 @@ fprintf(stderr, "%s: usb_control_msg() = %d\n", progname, res); usb_close(udev); + usleep(500000); } dev = find_dev(index); @@ -192,10 +189,6 @@ usb_dev_handle *udev = dev_open(dev); -// int res = usb_control_msg(udev, USB_TYPE_STANDARD | USB_ENDPOINT_IN, USB_REQ_GET_DESCRIPTOR, -// 0xfe, 0x0, buf, 0x0, 1000); -// fprintf(stderr, "%s: usb_control_msg() = %d\n", progname, res); - FILE *f = fopen(argv[1], "rb"); if (f == NULL) { fprintf(stderr, "%s: failed to open file '%s', exit.\n", progname, argv[1]); @@ -204,5 +197,16 @@ send_jpeg(f, udev); fclose(f); + while (1) { + char buf[2]; + int res = usb_control_msg(udev, USB_TYPE_VENDOR | USB_ENDPOINT_IN, USB_REQ_GET_DESCRIPTOR, + 0x0, 0x0, buf, 0x2, 1000); + fprintf(stderr, "%s: usb_control_msg() = %d\n", progname, res); + + usleep(500000); + } + + usb_close(udev); + return EXIT_SUCCESS; }
--- /dev/null Thu Jan 01 00:00:00 1970 +0000 +++ b/plug-mass_log.txt Sun May 08 18:22:22 2011 +0200 @@ -0,0 +1,67 @@ +ef67cb00 2793832298 S Ci:1:018:0 s 80 06 0100 0000 0012 18 < +ef67cb00 2793832615 C Ci:1:018:0 0 18 = 12010002 00000040 e8043520 00020102 0301 +deda9c80 2793832774 S Ci:1:018:0 s 80 06 0200 0000 0009 9 < +deda9c80 2793832985 C Ci:1:018:0 0 9 = 09022700 010100c0 32 +deda9c80 2793833055 S Ci:1:018:0 s 80 06 0200 0000 0027 39 < +deda9c80 2793833359 C Ci:1:018:0 0 39 = 09022700 010100c0 32090400 00030806 50000705 81020002 00070502 02000200 +ede01780 2794401395 S Ci:1:018:0 s 80 06 0100 0000 0040 64 < +ede01780 2794401603 C Ci:1:018:0 0 18 = 12010002 00000040 e8043520 00020102 0301 +ede01100 2794632908 S Ci:1:018:0 s 80 06 0100 0000 0012 18 < +ede01100 2794633102 C Ci:1:018:0 0 18 = 12010002 00000040 e8043520 00020102 0301 +ede01100 2794637808 S Ci:1:018:0 s 80 06 0200 0000 0009 9 < +ede01100 2794638105 C Ci:1:018:0 0 9 = 09022700 010100c0 32 +ede01780 2794654838 S Ci:1:018:0 s 80 06 0300 0000 00ff 255 < +ede01780 2794655104 C Ci:1:018:0 0 4 = 04030904 +ede01780 2794658803 S Ci:1:018:0 s 80 06 0303 0409 00ff 255 < +ede01780 2794658978 C Ci:1:018:0 0 14 = 0e034900 52003100 30000000 0000 +ede01780 2794663340 S Ci:1:018:0 s 80 06 0200 0000 00ff 255 < +ede01780 2794663604 C Ci:1:018:0 0 39 = 09022700 010100c0 32090400 00030806 50000705 81020002 00070502 02000200 +ede01780 2794671490 S Ci:1:018:0 s 80 06 0300 0000 00ff 255 < +ede01780 2794671736 C Ci:1:018:0 0 4 = 04030904 +ede01780 2794678269 S Ci:1:018:0 s 80 06 0302 0409 00ff 255 < +ede01780 2794678476 C Ci:1:018:0 0 56 = 38035300 41004d00 53005500 4e004700 20004400 69006700 69007400 61006c00 +ede01780 2794685397 S Ci:1:018:0 s 80 06 0300 0000 00ff 255 < +ede01780 2794685607 C Ci:1:018:0 0 4 = 04030904 +ede01780 2794693354 S Ci:1:018:0 s 80 06 0302 0409 00ff 255 < +ede01780 2794693606 C Ci:1:018:0 0 56 = 38035300 41004d00 53005500 4e004700 20004400 69006700 69007400 61006c00 +ede6a580 2794755583 S Ci:1:018:0 s 80 06 0100 0000 0012 18 < +ede6a580 2794755854 C Ci:1:018:0 0 18 = 12010002 00000040 e8043520 00020102 0301 +ede6a580 2794759604 S Ci:1:018:0 s 80 06 0200 0000 0009 9 < +ede6a580 2794759853 C Ci:1:018:0 0 9 = 09022700 010100c0 32 +ede6a580 2794764735 S Ci:1:018:0 s 80 06 0200 0000 0027 39 < +ede6a580 2794764980 C Ci:1:018:0 0 39 = 09022700 010100c0 32090400 00030806 50000705 81020002 00070502 02000200 +edf32180 2794769569 S Ci:1:018:0 s 80 06 0300 0000 0002 2 < +edf32180 2794769865 C Ci:1:018:0 0 2 = 0403 +edf32180 2794776538 S Ci:1:018:0 s 80 06 0300 0000 0004 4 < +edf32180 2794776730 C Ci:1:018:0 0 4 = 04030904 +ede6a580 2794783402 S Ci:1:018:0 s 80 06 0303 0409 0002 2 < +ede6a580 2794783599 C Ci:1:018:0 0 2 = 0e03 +edf32180 2794791041 S Ci:1:018:0 s 80 06 0303 0409 000e 14 < +edf32180 2794791238 C Ci:1:018:0 0 14 = 0e034900 52003100 30000000 0000 +edf32180 2794956855 S Ci:1:018:0 s a1 fe 0000 0000 0001 1 < +edf32180 2794957107 C Ci:1:018:0 0 1 = 01 +edf32180 2794998006 S Co:1:018:0 s 02 01 0000 0081 0000 0 +edf32180 2794998235 C Co:1:018:0 0 0 +dee8c000 2795024479 S Co:1:018:0 s 02 01 0000 0081 0000 0 +dee8c000 2795024731 C Co:1:018:0 0 0 +dee8c000 2795055506 S Co:1:018:0 s 02 01 0000 0081 0000 0 +dee8c000 2795055729 C Co:1:018:0 0 0 +e358db80 2795214780 S Co:1:018:0 s 02 01 0000 0081 0000 0 +e358db80 2795215117 C Co:1:018:0 0 0 +e358db80 2795236568 S Co:1:018:0 s 02 01 0000 0081 0000 0 +e358db80 2795236856 C Co:1:018:0 0 0 +e358db80 2795282524 S Co:1:018:0 s 02 01 0000 0081 0000 0 +e358db80 2795282864 C Co:1:018:0 0 0 +e358db80 2795304456 S Co:1:018:0 s 02 01 0000 0081 0000 0 +e358db80 2795304728 C Co:1:018:0 0 0 +e358d100 2795494769 S Ci:1:018:0 s 80 06 00fe 00fe 00ff 255 < +e358d100 2795499103 C Ci:1:018:0 0 8 = 00000001 a8019100 +e358d100 2795511395 S Ci:1:018:0 s 80 06 00fe 00fe 00e0 224 < +e358d100 2795511604 C Ci:1:018:0 0 0 +def69480 2795635976 S Ci:1:018:0 s 80 06 00fe 00fe 00fb 251 < +def69480 2795797513 C Ci:1:018:0 0 0 +e358dd00 2797809223 S Co:1:018:0 s 02 01 0000 0081 0000 0 +e358dd00 2797809494 C Co:1:018:0 0 0 +d4bd8e80 2797926292 S Co:1:018:0 s 02 01 0000 0081 0000 0 +d4bd8e80 2797926612 C Co:1:018:0 0 0 +
--- /dev/null Thu Jan 01 00:00:00 1970 +0000 +++ b/to-mass_log.txt Sun May 08 18:22:22 2011 +0200 @@ -0,0 +1,65 @@ +ef4c5e80 892738751 S Ci:1:014:0 s c0 06 0000 0000 0002 2 < +ef4c5e80 892738989 C Ci:1:014:0 0 2 = 0000 +ef4c5880 893263743 S Ci:1:014:0 s c0 06 0000 0000 0002 2 < +ef4c5880 893263990 C Ci:1:014:0 0 2 = 0000 +f0946200 893799754 S Ci:1:014:0 s c0 06 0000 0000 0002 2 < +f0946200 893799995 C Ci:1:014:0 0 2 = 0000 +def03200 894296627 S Ci:1:014:0 s c0 06 0000 0000 0002 2 < +def03200 894297004 C Ci:1:014:0 0 2 = 0000 +def03b80 894822781 S Ci:1:014:0 s c0 06 0000 0000 0002 2 < +def03b80 894822991 C Ci:1:014:0 0 2 = 0000 +ef515a80 895344445 S Co:1:014:0 s 40 05 0000 0000 0004 4 = a3394200 +ef515380 895344825 S Ci:1:001:0 s a3 00 0000 0001 0004 4 < +ef515380 895344839 C Ci:1:001:0 0 4 = 00010100 +ef515380 895344860 S Co:1:001:0 s 23 01 0010 0001 0000 0 +ef515380 895344868 C Co:1:001:0 0 0 +ef515a80 895345238 C Co:1:014:0 -108 4 > +ef515380 895345702 S Ci:1:001:0 s a3 00 0000 0001 0004 4 < +ef515380 895345713 C Ci:1:001:0 0 4 = 00010000 +edd94200 895374748 S Ci:1:001:0 s a3 00 0000 0001 0004 4 < +edd94200 895374764 C Ci:1:001:0 0 4 = 01050100 +edd94200 895374773 S Co:1:001:0 s 23 01 0010 0001 0000 0 +edd94200 895374781 C Co:1:001:0 0 0 +ef499780 895407629 S Ci:1:001:0 s a3 00 0000 0001 0004 4 < +ef499780 895407752 C Ci:1:001:0 0 4 = 01050000 +ef499780 895439749 S Ci:1:001:0 s a3 00 0000 0001 0004 4 < +ef499780 895439769 C Ci:1:001:0 0 4 = 01050000 +ef499780 895471727 S Ci:1:001:0 s a3 00 0000 0001 0004 4 < +ef499780 895471744 C Ci:1:001:0 0 4 = 01050000 +ef499d80 895503720 S Ci:1:001:0 s a3 00 0000 0001 0004 4 < +ef499d80 895503737 C Ci:1:001:0 0 4 = 01050000 +ef499d80 895503762 S Co:1:001:0 s 23 03 0004 0001 0000 0 +ef499d80 895503770 C Co:1:001:0 0 0 +ef499d80 895559772 S Ci:1:001:0 s a3 00 0000 0001 0004 4 < +ef499d80 895560045 C Ci:1:001:0 0 4 = 03051000 +ef499d80 895615722 S Co:1:001:0 s 23 01 0014 0001 0000 0 +ef499d80 895615739 C Co:1:001:0 0 0 +ef499d80 895615770 S Ci:1:000:0 s 80 06 0100 0000 0040 64 < +ef499d80 895615987 C Ci:1:000:0 0 18 = 12010002 00000040 e8043520 00020102 0301 +ef499d80 895616046 S Co:1:001:0 s 23 03 0004 0001 0000 0 +ef499d80 895616059 C Co:1:001:0 0 0 +edf32b00 895671760 S Ci:1:001:0 s a3 00 0000 0001 0004 4 < +edf32b00 895672100 C Ci:1:001:0 0 4 = 03051000 +edf32b00 895727746 S Co:1:001:0 s 23 01 0014 0001 0000 0 +edf32b00 895727766 C Co:1:001:0 0 0 +edf32b00 895727777 S Co:1:000:0 s 00 05 000f 0000 0000 0 +edf32b00 895727989 C Co:1:000:0 0 0 +edf32b00 895747746 S Ci:1:015:0 s 80 06 0100 0000 0012 18 < +edf32b00 895747987 C Ci:1:015:0 0 18 = 12010002 00000040 e8043520 00020102 0301 +eea43880 895748035 S Ci:1:015:0 s 80 06 0200 0000 0009 9 < +eea43880 895748239 C Ci:1:015:0 0 9 = 09022700 010100c0 32 +eea43880 895748311 S Ci:1:015:0 s 80 06 0200 0000 0027 39 < +eea43880 895748483 C Ci:1:015:0 0 39 = 09022700 010100c0 32090400 00030806 50000705 81020002 00070502 02000200 +eea43880 895748524 S Ci:1:015:0 s 80 06 0300 0000 00ff 255 < +eea43880 895748732 C Ci:1:015:0 0 4 = 04030904 +eea43880 895748757 S Ci:1:015:0 s 80 06 0302 0409 00ff 255 < +eea43880 895748987 C Ci:1:015:0 0 56 = 38035300 41004d00 53005500 4e004700 20004400 69006700 69007400 61006c00 +eea43880 895749065 S Ci:1:015:0 s 80 06 0301 0409 00ff 255 < +eea43880 895749233 C Ci:1:015:0 0 22 = 16035300 41004d00 53005500 4e004700 00000000 0000 +eea43880 895749274 S Ci:1:015:0 s 80 06 0303 0409 00ff 255 < +eea43880 895749487 C Ci:1:015:0 0 14 = 0e034900 52003100 30000000 0000 +eea43880 895749838 S Co:1:015:0 s 00 09 0001 0000 0000 0 +eea43880 895750113 C Co:1:015:0 0 0 +ede01c00 895771384 S Ci:1:001:0 s a3 00 0000 0001 0004 4 < +ede01c00 895771406 C Ci:1:001:0 0 4 = 03050000 +
--- /dev/null Thu Jan 01 00:00:00 1970 +0000 +++ b/to-mini_log.txt Sun May 08 18:22:22 2011 +0200 @@ -0,0 +1,55 @@ +f69d0a00 1137091779 S Ci:1:015:0 s 80 06 00fe 00fe 00fe 254 < +edc41c00 1137092082 S Ci:1:001:0 s a3 00 0000 0001 0004 4 < +edc41c00 1137092093 C Ci:1:001:0 0 4 = 00010100 +edc41c00 1137092106 S Co:1:001:0 s 23 01 0010 0001 0000 0 +edc41c00 1137092116 C Co:1:001:0 0 0 +f69d0a00 1137092773 C Ci:1:015:0 -108 0 +edc41800 1137098970 S Ci:1:001:0 s a3 00 0000 0001 0004 4 < +edc41800 1137099000 C Ci:1:001:0 0 4 = 00010000 +e24f9780 1137131957 S Ci:1:001:0 s a3 00 0000 0001 0004 4 < +e24f9780 1137131979 C Ci:1:001:0 0 4 = 01050100 +e24f9780 1137131989 S Co:1:001:0 s 23 01 0010 0001 0000 0 +e24f9780 1137131995 C Co:1:001:0 0 0 +dac9b300 1137163733 S Ci:1:001:0 s a3 00 0000 0001 0004 4 < +dac9b300 1137163753 C Ci:1:001:0 0 4 = 01050000 +dac9b300 1137195730 S Ci:1:001:0 s a3 00 0000 0001 0004 4 < +dac9b300 1137195751 C Ci:1:001:0 0 4 = 01050000 +dac9b300 1137227738 S Ci:1:001:0 s a3 00 0000 0001 0004 4 < +dac9b300 1137227758 C Ci:1:001:0 0 4 = 01050000 +dac9b300 1137259745 S Ci:1:001:0 s a3 00 0000 0001 0004 4 < +dac9b300 1137259764 C Ci:1:001:0 0 4 = 01050000 +dac9b300 1137259787 S Co:1:001:0 s 23 03 0004 0001 0000 0 +dac9b300 1137259795 C Co:1:001:0 0 0 +dec36180 1137315746 S Ci:1:001:0 s a3 00 0000 0001 0004 4 < +dec36180 1137316006 C Ci:1:001:0 0 4 = 03051000 +dec36180 1137372621 S Co:1:001:0 s 23 01 0014 0001 0000 0 +dec36180 1137372640 C Co:1:001:0 0 0 +dec36180 1137372673 S Ci:1:000:0 s 80 06 0100 0000 0040 64 < +dec36180 1137372893 C Ci:1:000:0 0 18 = 12010002 ff000040 e8043620 00020102 0301 +dec36180 1137373084 S Co:1:001:0 s 23 03 0004 0001 0000 0 +dec36180 1137373096 C Co:1:001:0 0 0 +dec36300 1137428918 S Ci:1:001:0 s a3 00 0000 0001 0004 4 < +dec36300 1137429143 C Ci:1:001:0 0 4 = 03051000 +eddfc800 1137486733 S Co:1:001:0 s 23 01 0014 0001 0000 0 +eddfc800 1137486752 C Co:1:001:0 0 0 +eddfc800 1137486765 S Co:1:000:0 s 00 05 0010 0000 0000 0 +eddfc800 1137487018 C Co:1:000:0 0 0 +eddfce80 1137506794 S Ci:1:016:0 s 80 06 0100 0000 0012 18 < +eddfce80 1137507017 C Ci:1:016:0 0 18 = 12010002 ff000040 e8043620 00020102 0301 +eddfce80 1137507065 S Ci:1:016:0 s 80 06 0200 0000 0009 9 < +eddfce80 1137507263 C Ci:1:016:0 0 9 = 09022700 010100c0 32 +eddfce80 1137507309 S Ci:1:016:0 s 80 06 0200 0000 0027 39 < +eddfce80 1137507515 C Ci:1:016:0 0 39 = 09022700 010100c0 32090400 0003ffff ff000705 81020002 00070502 02000200 +eddfce80 1137507558 S Ci:1:016:0 s 80 06 0300 0000 00ff 255 < +eddfce80 1137507765 C Ci:1:016:0 0 4 = 04030904 +eddfce80 1137507816 S Ci:1:016:0 s 80 06 0302 0409 00ff 255 < +eddfce80 1137508015 C Ci:1:016:0 0 56 = 38035300 41004d00 53005500 4e004700 20004400 69006700 69007400 61006c00 +eddfce80 1137508049 S Ci:1:016:0 s 80 06 0301 0409 00ff 255 < +eddfce80 1137508263 C Ci:1:016:0 0 22 = 16035300 41004d00 53005500 4e004700 00000000 0000 +eddfce80 1137508298 S Ci:1:016:0 s 80 06 0303 0409 00ff 255 < +eddfce80 1137508512 C Ci:1:016:0 0 14 = 0e034900 52003100 30000000 0000 +eddfce80 1137508872 S Co:1:016:0 s 00 09 0001 0000 0000 0 +eddfce80 1137509142 C Co:1:016:0 0 0 +eddfce80 1137509511 S Ci:1:001:0 s a3 00 0000 0001 0004 4 < +eddfce80 1137509526 C Ci:1:001:0 0 4 = 03050000 +
--- /dev/null Thu Jan 01 00:00:00 1970 +0000 +++ b/vfb/Makefile Sun May 08 18:22:22 2011 +0200 @@ -0,0 +1,5 @@ +obj-m := vfb.o +KDIR := /lib/modules/$(shell uname -r)/build +PWD := $(shell pwd) +default: + $(MAKE) -C $(KDIR) SUBDIRS=$(PWD) modules
--- /dev/null Thu Jan 01 00:00:00 1970 +0000 +++ b/vfb/vfb.c Sun May 08 18:22:22 2011 +0200 @@ -0,0 +1,610 @@ +/* + * linux/drivers/video/vfb.c -- Virtual frame buffer device + * + * Copyright (C) 2002 James Simmons + * + * Copyright (C) 1997 Geert Uytterhoeven + * + * This file is subject to the terms and conditions of the GNU General Public + * License. See the file COPYING in the main directory of this archive for + * more details. + */ + +#include <linux/module.h> +#include <linux/kernel.h> +#include <linux/errno.h> +#include <linux/string.h> +#include <linux/mm.h> +#include <linux/vmalloc.h> +#include <linux/delay.h> +#include <linux/interrupt.h> +#include <linux/platform_device.h> + +#include <linux/fb.h> +#include <linux/init.h> + + /* + * RAM we reserve for the frame buffer. This defines the maximum screen + * size + * + * The default can be overridden if the driver is compiled as a module + */ + +#define VIDEOMEMSIZE (1*1024*1024) /* 1 MB */ + +static void *videomemory; +static u_long videomemorysize = VIDEOMEMSIZE; +module_param(videomemorysize, ulong, 0); + +/********************************************************************** + * + * Memory management + * + **********************************************************************/ +static void *rvmalloc(unsigned long size) +{ + void *mem; + unsigned long adr; + + size = PAGE_ALIGN(size); + mem = vmalloc_32(size); + if (!mem) + return NULL; + + memset(mem, 0, size); /* Clear the ram out, no junk to the user */ + adr = (unsigned long) mem; + while (size > 0) { + SetPageReserved(vmalloc_to_page((void *)adr)); + adr += PAGE_SIZE; + size -= PAGE_SIZE; + } + + return mem; +} + +static void rvfree(void *mem, unsigned long size) +{ + unsigned long adr; + + if (!mem) + return; + + adr = (unsigned long) mem; + while ((long) size > 0) { + ClearPageReserved(vmalloc_to_page((void *)adr)); + adr += PAGE_SIZE; + size -= PAGE_SIZE; + } + vfree(mem); +} + +static struct fb_var_screeninfo vfb_default __devinitdata = { + .xres = 640, + .yres = 480, + .xres_virtual = 640, + .yres_virtual = 480, + .bits_per_pixel = 8, + .red = { 0, 8, 0 }, + .green = { 0, 8, 0 }, + .blue = { 0, 8, 0 }, + .activate = FB_ACTIVATE_TEST, + .height = -1, + .width = -1, + .pixclock = 20000, + .left_margin = 64, + .right_margin = 64, + .upper_margin = 32, + .lower_margin = 32, + .hsync_len = 64, + .vsync_len = 2, + .vmode = FB_VMODE_NONINTERLACED, +}; + +static struct fb_fix_screeninfo vfb_fix __devinitdata = { + .id = "Virtual FB", + .type = FB_TYPE_PACKED_PIXELS, + .visual = FB_VISUAL_PSEUDOCOLOR, + .xpanstep = 1, + .ypanstep = 1, + .ywrapstep = 1, + .accel = FB_ACCEL_NONE, +}; + +static int vfb_enable __initdata = 0; /* disabled by default */ +module_param(vfb_enable, bool, 0); + +static int vfb_check_var(struct fb_var_screeninfo *var, + struct fb_info *info); +static int vfb_set_par(struct fb_info *info); +static int vfb_setcolreg(u_int regno, u_int red, u_int green, u_int blue, + u_int transp, struct fb_info *info); +static int vfb_pan_display(struct fb_var_screeninfo *var, + struct fb_info *info); +static int vfb_mmap(struct fb_info *info, + struct vm_area_struct *vma); + +static struct fb_ops vfb_ops = { + .fb_read = fb_sys_read, + .fb_write = fb_sys_write, + .fb_check_var = vfb_check_var, + .fb_set_par = vfb_set_par, + .fb_setcolreg = vfb_setcolreg, + .fb_pan_display = vfb_pan_display, + .fb_fillrect = sys_fillrect, + .fb_copyarea = sys_copyarea, + .fb_imageblit = sys_imageblit, + .fb_mmap = vfb_mmap, +}; + + /* + * Internal routines + */ + +static u_long get_line_length(int xres_virtual, int bpp) +{ + u_long length; + + length = xres_virtual * bpp; + length = (length + 31) & ~31; + length >>= 3; + return (length); +} + + /* + * Setting the video mode has been split into two parts. + * First part, xxxfb_check_var, must not write anything + * to hardware, it should only verify and adjust var. + * This means it doesn't alter par but it does use hardware + * data from it to check this var. + */ + +static int vfb_check_var(struct fb_var_screeninfo *var, + struct fb_info *info) +{ + u_long line_length; + + /* + * FB_VMODE_CONUPDATE and FB_VMODE_SMOOTH_XPAN are equal! + * as FB_VMODE_SMOOTH_XPAN is only used internally + */ + + if (var->vmode & FB_VMODE_CONUPDATE) { + var->vmode |= FB_VMODE_YWRAP; + var->xoffset = info->var.xoffset; + var->yoffset = info->var.yoffset; + } + + /* + * Some very basic checks + */ + if (!var->xres) + var->xres = 1; + if (!var->yres) + var->yres = 1; + if (var->xres > var->xres_virtual) + var->xres_virtual = var->xres; + if (var->yres > var->yres_virtual) + var->yres_virtual = var->yres; + if (var->bits_per_pixel <= 1) + var->bits_per_pixel = 1; + else if (var->bits_per_pixel <= 8) + var->bits_per_pixel = 8; + else if (var->bits_per_pixel <= 16) + var->bits_per_pixel = 16; + else if (var->bits_per_pixel <= 24) + var->bits_per_pixel = 24; + else if (var->bits_per_pixel <= 32) + var->bits_per_pixel = 32; + else + return -EINVAL; + + if (var->xres_virtual < var->xoffset + var->xres) + var->xres_virtual = var->xoffset + var->xres; + if (var->yres_virtual < var->yoffset + var->yres) + var->yres_virtual = var->yoffset + var->yres; + + /* + * Memory limit + */ + line_length = + get_line_length(var->xres_virtual, var->bits_per_pixel); + if (line_length * var->yres_virtual > videomemorysize) + return -ENOMEM; + + /* + * Now that we checked it we alter var. The reason being is that the video + * mode passed in might not work but slight changes to it might make it + * work. This way we let the user know what is acceptable. + */ + switch (var->bits_per_pixel) { + case 1: + case 8: + var->red.offset = 0; + var->red.length = 8; + var->green.offset = 0; + var->green.length = 8; + var->blue.offset = 0; + var->blue.length = 8; + var->transp.offset = 0; + var->transp.length = 0; + break; + case 16: /* RGBA 5551 */ + if (var->transp.length) { + var->red.offset = 0; + var->red.length = 5; + var->green.offset = 5; + var->green.length = 5; + var->blue.offset = 10; + var->blue.length = 5; + var->transp.offset = 15; + var->transp.length = 1; + } else { /* RGB 565 */ + var->red.offset = 0; + var->red.length = 5; + var->green.offset = 5; + var->green.length = 6; + var->blue.offset = 11; + var->blue.length = 5; + var->transp.offset = 0; + var->transp.length = 0; + } + break; + case 24: /* RGB 888 */ + var->red.offset = 0; + var->red.length = 8; + var->green.offset = 8; + var->green.length = 8; + var->blue.offset = 16; + var->blue.length = 8; + var->transp.offset = 0; + var->transp.length = 0; + break; + case 32: /* RGBA 8888 */ + var->red.offset = 0; + var->red.length = 8; + var->green.offset = 8; + var->green.length = 8; + var->blue.offset = 16; + var->blue.length = 8; + var->transp.offset = 24; + var->transp.length = 8; + break; + } + var->red.msb_right = 0; + var->green.msb_right = 0; + var->blue.msb_right = 0; + var->transp.msb_right = 0; + + return 0; +} + +/* This routine actually sets the video mode. It's in here where we + * the hardware state info->par and fix which can be affected by the + * change in par. For this driver it doesn't do much. + */ +static int vfb_set_par(struct fb_info *info) +{ + info->fix.line_length = get_line_length(info->var.xres_virtual, + info->var.bits_per_pixel); + return 0; +} + + /* + * Set a single color register. The values supplied are already + * rounded down to the hardware's capabilities (according to the + * entries in the var structure). Return != 0 for invalid regno. + */ + +static int vfb_setcolreg(u_int regno, u_int red, u_int green, u_int blue, + u_int transp, struct fb_info *info) +{ + if (regno >= 256) /* no. of hw registers */ + return 1; + /* + * Program hardware... do anything you want with transp + */ + + /* grayscale works only partially under directcolor */ + if (info->var.grayscale) { + /* grayscale = 0.30*R + 0.59*G + 0.11*B */ + red = green = blue = + (red * 77 + green * 151 + blue * 28) >> 8; + } + + /* Directcolor: + * var->{color}.offset contains start of bitfield + * var->{color}.length contains length of bitfield + * {hardwarespecific} contains width of RAMDAC + * cmap[X] is programmed to (X << red.offset) | (X << green.offset) | (X << blue.offset) + * RAMDAC[X] is programmed to (red, green, blue) + * + * Pseudocolor: + * var->{color}.offset is 0 unless the palette index takes less than + * bits_per_pixel bits and is stored in the upper + * bits of the pixel value + * var->{color}.length is set so that 1 << length is the number of available + * palette entries + * cmap is not used + * RAMDAC[X] is programmed to (red, green, blue) + * + * Truecolor: + * does not use DAC. Usually 3 are present. + * var->{color}.offset contains start of bitfield + * var->{color}.length contains length of bitfield + * cmap is programmed to (red << red.offset) | (green << green.offset) | + * (blue << blue.offset) | (transp << transp.offset) + * RAMDAC does not exist + */ +#define CNVT_TOHW(val,width) ((((val)<<(width))+0x7FFF-(val))>>16) + switch (info->fix.visual) { + case FB_VISUAL_TRUECOLOR: + case FB_VISUAL_PSEUDOCOLOR: + red = CNVT_TOHW(red, info->var.red.length); + green = CNVT_TOHW(green, info->var.green.length); + blue = CNVT_TOHW(blue, info->var.blue.length); + transp = CNVT_TOHW(transp, info->var.transp.length); + break; + case FB_VISUAL_DIRECTCOLOR: + red = CNVT_TOHW(red, 8); /* expect 8 bit DAC */ + green = CNVT_TOHW(green, 8); + blue = CNVT_TOHW(blue, 8); + /* hey, there is bug in transp handling... */ + transp = CNVT_TOHW(transp, 8); + break; + } +#undef CNVT_TOHW + /* Truecolor has hardware independent palette */ + if (info->fix.visual == FB_VISUAL_TRUECOLOR) { + u32 v; + + if (regno >= 16) + return 1; + + v = (red << info->var.red.offset) | + (green << info->var.green.offset) | + (blue << info->var.blue.offset) | + (transp << info->var.transp.offset); + switch (info->var.bits_per_pixel) { + case 8: + break; + case 16: + ((u32 *) (info->pseudo_palette))[regno] = v; + break; + case 24: + case 32: + ((u32 *) (info->pseudo_palette))[regno] = v; + break; + } + return 0; + } + return 0; +} + + /* + * Pan or Wrap the Display + * + * This call looks only at xoffset, yoffset and the FB_VMODE_YWRAP flag + */ + +static int vfb_pan_display(struct fb_var_screeninfo *var, + struct fb_info *info) +{ + if (var->vmode & FB_VMODE_YWRAP) { + if (var->yoffset < 0 + || var->yoffset >= info->var.yres_virtual + || var->xoffset) + return -EINVAL; + } else { + if (var->xoffset + var->xres > info->var.xres_virtual || + var->yoffset + var->yres > info->var.yres_virtual) + return -EINVAL; + } + info->var.xoffset = var->xoffset; + info->var.yoffset = var->yoffset; + if (var->vmode & FB_VMODE_YWRAP) + info->var.vmode |= FB_VMODE_YWRAP; + else + info->var.vmode &= ~FB_VMODE_YWRAP; + return 0; +} + + /* + * Most drivers don't need their own mmap function + */ + +static int vfb_mmap(struct fb_info *info, + struct vm_area_struct *vma) +{ + unsigned long start = vma->vm_start; + unsigned long size = vma->vm_end - vma->vm_start; + unsigned long offset = vma->vm_pgoff << PAGE_SHIFT; + unsigned long page, pos; + + if (offset + size > info->fix.smem_len) { + return -EINVAL; + } + + pos = (unsigned long)info->fix.smem_start + offset; + + while (size > 0) { + page = vmalloc_to_pfn((void *)pos); + if (remap_pfn_range(vma, start, page, PAGE_SIZE, PAGE_SHARED)) { + return -EAGAIN; + } + start += PAGE_SIZE; + pos += PAGE_SIZE; + if (size > PAGE_SIZE) + size -= PAGE_SIZE; + else + size = 0; + } + + vma->vm_flags |= VM_RESERVED; /* avoid to swap out this VMA */ + return 0; + +} + +#ifndef MODULE +/* + * The virtual framebuffer driver is only enabled if explicitly + * requested by passing 'video=vfb:' (or any actual options). + */ +static int __init vfb_setup(char *options) +{ + char *this_opt; + + vfb_enable = 0; + + if (!options) + return 1; + + vfb_enable = 1; + + if (!*options) + return 1; + + while ((this_opt = strsep(&options, ",")) != NULL) { + if (!*this_opt) + continue; + /* Test disable for backwards compatibility */ + if (!strcmp(this_opt, "disable")) + vfb_enable = 0; + } + return 1; +} +#endif /* MODULE */ + + /* + * Initialisation + */ + +static int __devinit vfb_probe(struct platform_device *dev) +{ + struct fb_info *info; + int retval = -ENOMEM; + + /* + * For real video cards we use ioremap. + */ + if (!(videomemory = rvmalloc(videomemorysize))) + return retval; + + /* + * VFB must clear memory to prevent kernel info + * leakage into userspace + * VGA-based drivers MUST NOT clear memory if + * they want to be able to take over vgacon + */ + memset(videomemory, 0, videomemorysize); + + info = framebuffer_alloc(sizeof(u32) * 256, &dev->dev); + if (!info) + goto err; + + info->screen_base = (char __iomem *)videomemory; + info->fbops = &vfb_ops; + + retval = fb_find_mode(&info->var, info, NULL, + NULL, 0, NULL, 8); + + if (!retval || (retval == 4)) + info->var = vfb_default; + vfb_fix.smem_start = (unsigned long) videomemory; + vfb_fix.smem_len = videomemorysize; + info->fix = vfb_fix; + info->pseudo_palette = info->par; + info->par = NULL; + info->flags = FBINFO_FLAG_DEFAULT; + + retval = fb_alloc_cmap(&info->cmap, 256, 0); + if (retval < 0) + goto err1; + + retval = register_framebuffer(info); + if (retval < 0) + goto err2; + platform_set_drvdata(dev, info); + + printk(KERN_INFO + "fb%d: Virtual frame buffer device, using %ldK of video memory\n", + info->node, videomemorysize >> 10); + return 0; +err2: + fb_dealloc_cmap(&info->cmap); +err1: + framebuffer_release(info); +err: + rvfree(videomemory, videomemorysize); + return retval; +} + +static int vfb_remove(struct platform_device *dev) +{ + struct fb_info *info = platform_get_drvdata(dev); + + if (info) { + unregister_framebuffer(info); + rvfree(videomemory, videomemorysize); + fb_dealloc_cmap(&info->cmap); + framebuffer_release(info); + } + return 0; +} + +static struct platform_driver vfb_driver = { + .probe = vfb_probe, + .remove = vfb_remove, + .driver = { + .name = "vfb", + }, +}; + +static struct platform_device *vfb_device; + +static int __init vfb_init(void) +{ + int ret = 0; + +#ifndef MODULE + char *option = NULL; + + if (fb_get_options("vfb", &option)) + return -ENODEV; + vfb_setup(option); +#endif + + if (!vfb_enable) + return -ENXIO; + + ret = platform_driver_register(&vfb_driver); + + if (!ret) { + vfb_device = platform_device_alloc("vfb", 0); + + if (vfb_device) + ret = platform_device_add(vfb_device); + else + ret = -ENOMEM; + + if (ret) { + platform_device_put(vfb_device); + platform_driver_unregister(&vfb_driver); + } + } + + return ret; +} + +module_init(vfb_init); + +#ifdef MODULE +static void __exit vfb_exit(void) +{ + platform_device_unregister(vfb_device); + platform_driver_unregister(&vfb_driver); +} + +module_exit(vfb_exit); + +MODULE_LICENSE("GPL"); +#endif /* MODULE */