All of lore.kernel.org
 help / color / mirror / Atom feed
From: Michal Simek <monstr@monstr.eu>
To: Stefan Kristiansson <stefan.kristiansson@saunalahti.fi>
Cc: linux-kernel@vger.kernel.org, linux-fbdev@vger.kernel.org,
	tomi.valkeinen@ti.com, plagnioj@jcrosoft.com
Subject: Re: [PATCH v3] video: add OpenCores VGA/LCD framebuffer driver
Date: Tue, 26 Nov 2013 07:10:46 +0000	[thread overview]
Message-ID: <52944976.4060003@monstr.eu> (raw)
In-Reply-To: <1385094870-6962-1-git-send-email-stefan.kristiansson@saunalahti.fi>

[-- Attachment #1: Type: text/plain, Size: 13267 bytes --]

On 11/22/2013 05:34 AM, Stefan Kristiansson wrote:
> This adds support for the VGA/LCD core available from OpenCores:
> http://opencores.org/project,vga_lcd
> 
> The driver have been tested together with both OpenRISC and
> ARM (socfpga) processors.
> 
> Signed-off-by: Stefan Kristiansson <stefan.kristiansson@saunalahti.fi>
> ---
> Changes in v2:
> - Add Microblaze as an example user and fix a typo in Xilinx Zynq
> 
> Changes in v3:
> - Use devm_kzalloc instead of kzalloc
> - Remove superflous MODULE #ifdef
> ---
>  drivers/video/Kconfig  |  17 ++
>  drivers/video/Makefile |   1 +
>  drivers/video/ocfb.c   | 471 +++++++++++++++++++++++++++++++++++++++++++++++++
>  3 files changed, 489 insertions(+)
>  create mode 100644 drivers/video/ocfb.c
> 
> diff --git a/drivers/video/Kconfig b/drivers/video/Kconfig
> index 84b685f..3b3f31e 100644
> --- a/drivers/video/Kconfig
> +++ b/drivers/video/Kconfig
> @@ -979,6 +979,23 @@ config FB_PVR2
>  	  (<file:drivers/video/pvr2fb.c>). Please see the file
>  	  <file:Documentation/fb/pvr2fb.txt>.
>  
> +config FB_OPENCORES
> +	tristate "OpenCores VGA/LCD core 2.0 framebuffer support"
> +	depends on FB
> +	select FB_CFB_FILLRECT
> +	select FB_CFB_COPYAREA
> +	select FB_CFB_IMAGEBLIT
> +	default n
> +	help
> +	  This enables support for the OpenCores VGA/LCD core.
> +
> +	  The OpenCores VGA/LCD core is typically used together with
> +	  softcore CPUs (e.g. OpenRISC or Microblaze) or hard processor
> +	  systems (e.g. Altera socfpga or Xilinx Zynq) on FPGAs.
> +
> +	  The source code and specification for the core is available at
> +	  <http://opencores.org/project,vga_lcd>
> +
>  config FB_S1D13XXX
>  	tristate "Epson S1D13XXX framebuffer support"
>  	depends on FB
> diff --git a/drivers/video/Makefile b/drivers/video/Makefile
> index e8bae8d..ae17ddf 100644
> --- a/drivers/video/Makefile
> +++ b/drivers/video/Makefile
> @@ -150,6 +150,7 @@ obj-$(CONFIG_FB_NUC900)           += nuc900fb.o
>  obj-$(CONFIG_FB_JZ4740)		  += jz4740_fb.o
>  obj-$(CONFIG_FB_PUV3_UNIGFX)      += fb-puv3.o
>  obj-$(CONFIG_FB_HYPERV)		  += hyperv_fb.o
> +obj-$(CONFIG_FB_OPENCORES)	  += ocfb.o
>  
>  # Platform or fallback drivers go here
>  obj-$(CONFIG_FB_UVESA)            += uvesafb.o
> diff --git a/drivers/video/ocfb.c b/drivers/video/ocfb.c
> new file mode 100644
> index 0000000..3bf5f67
> --- /dev/null
> +++ b/drivers/video/ocfb.c
> @@ -0,0 +1,471 @@
> +/*
> + * OpenCores VGA/LCD 2.0 core frame buffer driver
> + *
> + * Copyright (C) 2013 Stefan Kristiansson, stefan.kristiansson@saunalahti.fi
> + *
> + * This file is licensed under the terms of the GNU General Public License
> + * version 2.  This program is licensed "as is" without any warranty of any
> + * kind, whether express or implied.
> + */
> +
> +#include <linux/module.h>
> +#include <linux/kernel.h>
> +#include <linux/errno.h>
> +#include <linux/string.h>
> +#include <linux/slab.h>
> +#include <linux/delay.h>
> +#include <linux/mm.h>
> +#include <linux/dma-mapping.h>
> +#include <linux/fb.h>
> +#include <linux/init.h>
> +#include <linux/io.h>
> +#include <linux/platform_device.h>
> +#include <linux/of.h>
> +
> +/* OCFB register defines */
> +#define OCFB_CTRL	0x000
> +#define OCFB_STAT	0x004
> +#define OCFB_HTIM	0x008
> +#define OCFB_VTIM	0x00c
> +#define OCFB_HVLEN	0x010
> +#define OCFB_VBARA	0x014
> +#define OCFB_PALETTE	0x800
> +
> +#define OCFB_CTRL_VEN	0x00000001 /* Video Enable */
> +#define OCFB_CTRL_HIE	0x00000002 /* HSync Interrupt Enable */
> +#define OCFB_CTRL_PC	0x00000800 /* 8-bit Pseudo Color Enable*/
> +#define OCFB_CTRL_CD8	0x00000000 /* Color Depth 8 */
> +#define OCFB_CTRL_CD16	0x00000200 /* Color Depth 16 */
> +#define OCFB_CTRL_CD24	0x00000400 /* Color Depth 24 */
> +#define OCFB_CTRL_CD32	0x00000600 /* Color Depth 32 */
> +#define OCFB_CTRL_VBL1	0x00000000 /* Burst Length 1 */
> +#define OCFB_CTRL_VBL2	0x00000080 /* Burst Length 2 */
> +#define OCFB_CTRL_VBL4	0x00000100 /* Burst Length 4 */
> +#define OCFB_CTRL_VBL8	0x00000180 /* Burst Length 8 */
> +
> +#define PALETTE_SIZE	256
> +
> +#define OCFB_NAME	"OC VGA/LCD"
> +
> +static char *mode_option;
> +
> +static const struct fb_videomode default_mode = {
> +	/* 640x480 @ 60 Hz, 31.5 kHz hsync */
> +	NULL, 60, 640, 480, 39721, 40, 24, 32, 11, 96, 2,
> +	0, FB_VMODE_NONINTERLACED
> +};
> +
> +struct ocfb_dev {
> +	struct fb_info info;
> +	/* Physical and virtual addresses of control regs */
> +	phys_addr_t regs_phys;
> +	int regs_phys_size;
> +	void __iomem *regs;
> +	/* flag indicating whether the regs are little endian accessed */
> +	int little_endian;
> +	/* Physical and virtual addresses of framebuffer */
> +	phys_addr_t fb_phys;
> +	void __iomem *fb_virt;
> +	u32 pseudo_palette[PALETTE_SIZE];
> +};
> +
> +struct ocfb_par {
> +	void __iomem *pal_adr;
> +};
> +
> +static struct ocfb_par ocfb_par_priv;
> +
> +static struct fb_var_screeninfo ocfb_var;
> +static struct fb_fix_screeninfo ocfb_fix;
> +
> +#ifndef MODULE
> +static int __init ocfb_setup(char *options)
> +{
> +	char *curr_opt;
> +
> +	if (!options || !*options)
> +		return 0;
> +
> +	while ((curr_opt = strsep(&options, ",")) != NULL) {
> +		if (!*curr_opt)
> +			continue;
> +		mode_option = curr_opt;
> +	}
> +
> +	return 0;
> +}
> +#endif
> +
> +static inline u32 ocfb_readreg(struct ocfb_dev *fbdev, loff_t offset)
> +{
> +	if (fbdev->little_endian)
> +		return ioread32(fbdev->regs + offset);
> +	else
> +		return ioread32be(fbdev->regs + offset);
> +}
> +
> +static void ocfb_writereg(struct ocfb_dev *fbdev, loff_t offset, u32 data)
> +{
> +	if (fbdev->little_endian)
> +		iowrite32(data, fbdev->regs + offset);
> +	else
> +		iowrite32be(data, fbdev->regs + offset);
> +}
> +
> +static int ocfb_setupfb(struct ocfb_dev *fbdev)
> +{
> +	unsigned long bpp_config;
> +	struct fb_var_screeninfo *var = &fbdev->info.var;
> +	struct device *dev = fbdev->info.device;
> +	u32 hlen;
> +	u32 vlen;
> +
> +	/* Disable display */
> +	ocfb_writereg(fbdev, OCFB_CTRL, 0);
> +
> +	/* Register framebuffer address */
> +	fbdev->little_endian = 0;
> +	ocfb_writereg(fbdev, OCFB_VBARA, fbdev->fb_phys);
> +
> +	/* Detect endianess */
> +	if (ocfb_readreg(fbdev, OCFB_VBARA) != fbdev->fb_phys) {
> +		fbdev->little_endian = 1;
> +		ocfb_writereg(fbdev, OCFB_VBARA, fbdev->fb_phys);
> +	}
> +
> +	/* Horizontal timings */
> +	ocfb_writereg(fbdev, OCFB_HTIM, (var->hsync_len - 1) << 24 |
> +		      (var->right_margin - 1) << 16 | (var->xres - 1));
> +
> +	/* Vertical timings */
> +	ocfb_writereg(fbdev, OCFB_VTIM, (var->vsync_len - 1) << 24 |
> +		      (var->lower_margin - 1) << 16 | (var->yres - 1));
> +
> +	/* Total length of frame */
> +	hlen = var->left_margin + var->right_margin + var->hsync_len +
> +		var->xres;
> +
> +	vlen = var->upper_margin + var->lower_margin + var->vsync_len +
> +		var->yres;
> +
> +	ocfb_writereg(fbdev, OCFB_HVLEN, (hlen - 1) << 16 | (vlen - 1));
> +
> +	bpp_config = OCFB_CTRL_CD8;
> +	switch (var->bits_per_pixel) {
> +	case 8:
> +		if (!var->grayscale)
> +			bpp_config |= OCFB_CTRL_PC;  /* enable palette */
> +		break;
> +
> +	case 16:
> +		bpp_config |= OCFB_CTRL_CD16;
> +		break;
> +
> +	case 24:
> +		bpp_config |= OCFB_CTRL_CD24;
> +		break;
> +
> +	case 32:
> +		bpp_config |= OCFB_CTRL_CD32;
> +		break;
> +
> +	default:
> +		dev_err(dev, "no bpp specified\n");
> +		break;
> +	}
> +
> +	/* maximum (8) VBL (video memory burst length) */
> +	bpp_config |= OCFB_CTRL_VBL8;
> +
> +	/* Enable output */
> +	ocfb_writereg(fbdev, OCFB_CTRL, (OCFB_CTRL_VEN | bpp_config));
> +
> +	return 0;
> +}
> +
> +static int ocfb_setcolreg(unsigned regno, unsigned red, unsigned green,
> +			  unsigned blue, unsigned transp,
> +			  struct fb_info *info)
> +{
> +	struct ocfb_par *par = (struct ocfb_par *)info->par;
> +	u32 color;
> +
> +	if (regno >= info->cmap.len) {
> +		dev_err(info->device, "regno >= cmap.len\n");
> +		return 1;
> +	}
> +
> +	if (info->var.grayscale) {
> +		/* grayscale = 0.30*R + 0.59*G + 0.11*B */
> +		red = green = blue = (red * 77 + green * 151 + blue * 28) >> 8;
> +	}
> +
> +	red >>= (16 - info->var.red.length);
> +	green >>= (16 - info->var.green.length);
> +	blue >>= (16 - info->var.blue.length);
> +	transp >>= (16 - info->var.transp.length);
> +
> +	if (info->var.bits_per_pixel == 8 && !info->var.grayscale) {
> +		regno <<= 2;
> +		color = (red << 16) | (green << 8) | blue;
> +		ocfb_writereg(par->pal_adr, regno, color);
> +	} else {
> +		((u32 *)(info->pseudo_palette))[regno] =
> +			(red << info->var.red.offset) |
> +			(green << info->var.green.offset) |
> +			(blue << info->var.blue.offset) |
> +			(transp << info->var.transp.offset);
> +	}
> +
> +	return 0;
> +}
> +
> +static int ocfb_init_fix(struct ocfb_dev *fbdev)
> +{
> +	struct fb_var_screeninfo *var = &fbdev->info.var;
> +	struct fb_fix_screeninfo *fix = &fbdev->info.fix;
> +
> +	strcpy(fix->id, OCFB_NAME);
> +
> +	fix->line_length = var->xres * var->bits_per_pixel/8;
> +	fix->smem_len = fix->line_length * var->yres;
> +	fix->type = FB_TYPE_PACKED_PIXELS;
> +
> +	if (var->bits_per_pixel == 8 && !var->grayscale)
> +		fix->visual = FB_VISUAL_PSEUDOCOLOR;
> +	else
> +		fix->visual = FB_VISUAL_TRUECOLOR;
> +
> +	return 0;
> +}
> +
> +static int ocfb_init_var(struct ocfb_dev *fbdev)
> +{
> +	struct fb_var_screeninfo *var = &fbdev->info.var;
> +
> +	var->accel_flags = FB_ACCEL_NONE;
> +	var->activate = FB_ACTIVATE_NOW;
> +	var->xres_virtual = var->xres;
> +	var->yres_virtual = var->yres;
> +
> +	switch (var->bits_per_pixel) {
> +	case 8:
> +		var->transp.offset = 0;
> +		var->transp.length = 0;
> +		var->red.offset = 0;
> +		var->red.length = 8;
> +		var->green.offset = 0;
> +		var->green.length = 8;
> +		var->blue.offset = 0;
> +		var->blue.length = 8;
> +		break;
> +
> +	case 16:
> +		var->transp.offset = 0;
> +		var->transp.length = 0;
> +		var->red.offset = 11;
> +		var->red.length = 5;
> +		var->green.offset = 5;
> +		var->green.length = 6;
> +		var->blue.offset = 0;
> +		var->blue.length  = 5;
> +		break;
> +
> +	case 24:
> +		var->transp.offset = 0;
> +		var->transp.length = 0;
> +		var->red.offset = 16;
> +		var->red.length = 8;
> +		var->green.offset = 8;
> +		var->green.length = 8;
> +		var->blue.offset = 0;
> +		var->blue.length = 8;
> +		break;
> +
> +	case 32:
> +		var->transp.offset = 24;
> +		var->transp.length = 8;
> +		var->red.offset = 16;
> +		var->red.length = 8;
> +		var->green.offset = 8;
> +		var->green.length = 8;
> +		var->blue.offset = 0;
> +		var->blue.length = 8;
> +		break;
> +	}
> +
> +	return 0;
> +}
> +
> +static struct fb_ops ocfb_ops = {
> +	.owner		= THIS_MODULE,
> +	.fb_setcolreg	= ocfb_setcolreg,
> +	.fb_fillrect	= cfb_fillrect,
> +	.fb_copyarea	= cfb_copyarea,
> +	.fb_imageblit	= cfb_imageblit,
> +};
> +
> +static int ocfb_probe(struct platform_device *pdev)
> +{
> +	int ret = 0;
> +	struct ocfb_dev *fbdev;
> +	struct ocfb_par *par = &ocfb_par_priv;
> +	struct resource *res;
> +	struct resource *mmio;
> +	int fbsize;
> +
> +	fbdev = devm_kzalloc(&pdev->dev, sizeof(*fbdev), GFP_KERNEL);
> +	if (!fbdev)
> +		return -ENOMEM;
> +
> +	platform_set_drvdata(pdev, fbdev);
> +
> +	fbdev->info.fbops = &ocfb_ops;
> +	fbdev->info.var = ocfb_var;
> +	fbdev->info.fix = ocfb_fix;
> +	fbdev->info.device = &pdev->dev;
> +	fbdev->info.par = par;
> +
> +	/* Video mode setup */
> +	if (!fb_find_mode(&fbdev->info.var, &fbdev->info, mode_option,
> +			  NULL, 0, &default_mode, 16)) {
> +		dev_err(&pdev->dev, "No valid video modes found\n");
> +		return -EINVAL;
> +	}
> +	ocfb_init_var(fbdev);
> +	ocfb_init_fix(fbdev);
> +
> +	/* Request I/O resource */
> +	res = platform_get_resource(pdev, IORESOURCE_MEM, 0);
> +	if (!res) {
> +		dev_err(&pdev->dev, "I/O resource request failed\n");
> +		return -ENXIO;
> +	}
> +	fbdev->regs_phys = res->start;
> +	fbdev->regs_phys_size = resource_size(res);
> +	mmio = devm_request_mem_region(&pdev->dev, res->start,
> +				       resource_size(res), res->name);
> +	if (!mmio) {
> +		dev_err(&pdev->dev, "I/O memory space request failed\n");
> +		return -ENXIO;
> +	}
> +	fbdev->regs = devm_ioremap_nocache(&pdev->dev, mmio->start,
> +					   resource_size(mmio));
> +	if (!fbdev->regs) {
> +		dev_err(&pdev->dev, "I/O memory remap request failed\n");
> +		return -ENXIO;
> +	}


This construction seems to me too complicated and can be simpler.

What about this?
	res = platform_get_resource(pdev, IORESOURCE_MEM, 0);
	res->flags &= ~IORESOURCE_CACHEABLE;
	fbdev->regs = devm_ioremap_resource(&pdev->dev, res);
	if (IS_ERR(fbdev->regs))
		return PTR_ERR(fbdev->regs);

Thanks,
Michal


-- 
Michal Simek, Ing. (M.Eng), OpenPGP -> KeyID: FE3D1F91
w: www.monstr.eu p: +42-0-721842854
Maintainer of Linux kernel - Microblaze cpu - http://www.monstr.eu/fdt/
Maintainer of Linux kernel - Xilinx Zynq ARM architecture
Microblaze U-BOOT custodian and responsible for u-boot arm zynq platform



[-- Attachment #2: OpenPGP digital signature --]
[-- Type: application/pgp-signature, Size: 263 bytes --]

WARNING: multiple messages have this Message-ID (diff)
From: Michal Simek <monstr@monstr.eu>
To: Stefan Kristiansson <stefan.kristiansson@saunalahti.fi>
Cc: linux-kernel@vger.kernel.org, linux-fbdev@vger.kernel.org,
	tomi.valkeinen@ti.com, plagnioj@jcrosoft.com
Subject: Re: [PATCH v3] video: add OpenCores VGA/LCD framebuffer driver
Date: Tue, 26 Nov 2013 08:10:46 +0100	[thread overview]
Message-ID: <52944976.4060003@monstr.eu> (raw)
In-Reply-To: <1385094870-6962-1-git-send-email-stefan.kristiansson@saunalahti.fi>

[-- Attachment #1: Type: text/plain, Size: 13267 bytes --]

On 11/22/2013 05:34 AM, Stefan Kristiansson wrote:
> This adds support for the VGA/LCD core available from OpenCores:
> http://opencores.org/project,vga_lcd
> 
> The driver have been tested together with both OpenRISC and
> ARM (socfpga) processors.
> 
> Signed-off-by: Stefan Kristiansson <stefan.kristiansson@saunalahti.fi>
> ---
> Changes in v2:
> - Add Microblaze as an example user and fix a typo in Xilinx Zynq
> 
> Changes in v3:
> - Use devm_kzalloc instead of kzalloc
> - Remove superflous MODULE #ifdef
> ---
>  drivers/video/Kconfig  |  17 ++
>  drivers/video/Makefile |   1 +
>  drivers/video/ocfb.c   | 471 +++++++++++++++++++++++++++++++++++++++++++++++++
>  3 files changed, 489 insertions(+)
>  create mode 100644 drivers/video/ocfb.c
> 
> diff --git a/drivers/video/Kconfig b/drivers/video/Kconfig
> index 84b685f..3b3f31e 100644
> --- a/drivers/video/Kconfig
> +++ b/drivers/video/Kconfig
> @@ -979,6 +979,23 @@ config FB_PVR2
>  	  (<file:drivers/video/pvr2fb.c>). Please see the file
>  	  <file:Documentation/fb/pvr2fb.txt>.
>  
> +config FB_OPENCORES
> +	tristate "OpenCores VGA/LCD core 2.0 framebuffer support"
> +	depends on FB
> +	select FB_CFB_FILLRECT
> +	select FB_CFB_COPYAREA
> +	select FB_CFB_IMAGEBLIT
> +	default n
> +	help
> +	  This enables support for the OpenCores VGA/LCD core.
> +
> +	  The OpenCores VGA/LCD core is typically used together with
> +	  softcore CPUs (e.g. OpenRISC or Microblaze) or hard processor
> +	  systems (e.g. Altera socfpga or Xilinx Zynq) on FPGAs.
> +
> +	  The source code and specification for the core is available at
> +	  <http://opencores.org/project,vga_lcd>
> +
>  config FB_S1D13XXX
>  	tristate "Epson S1D13XXX framebuffer support"
>  	depends on FB
> diff --git a/drivers/video/Makefile b/drivers/video/Makefile
> index e8bae8d..ae17ddf 100644
> --- a/drivers/video/Makefile
> +++ b/drivers/video/Makefile
> @@ -150,6 +150,7 @@ obj-$(CONFIG_FB_NUC900)           += nuc900fb.o
>  obj-$(CONFIG_FB_JZ4740)		  += jz4740_fb.o
>  obj-$(CONFIG_FB_PUV3_UNIGFX)      += fb-puv3.o
>  obj-$(CONFIG_FB_HYPERV)		  += hyperv_fb.o
> +obj-$(CONFIG_FB_OPENCORES)	  += ocfb.o
>  
>  # Platform or fallback drivers go here
>  obj-$(CONFIG_FB_UVESA)            += uvesafb.o
> diff --git a/drivers/video/ocfb.c b/drivers/video/ocfb.c
> new file mode 100644
> index 0000000..3bf5f67
> --- /dev/null
> +++ b/drivers/video/ocfb.c
> @@ -0,0 +1,471 @@
> +/*
> + * OpenCores VGA/LCD 2.0 core frame buffer driver
> + *
> + * Copyright (C) 2013 Stefan Kristiansson, stefan.kristiansson@saunalahti.fi
> + *
> + * This file is licensed under the terms of the GNU General Public License
> + * version 2.  This program is licensed "as is" without any warranty of any
> + * kind, whether express or implied.
> + */
> +
> +#include <linux/module.h>
> +#include <linux/kernel.h>
> +#include <linux/errno.h>
> +#include <linux/string.h>
> +#include <linux/slab.h>
> +#include <linux/delay.h>
> +#include <linux/mm.h>
> +#include <linux/dma-mapping.h>
> +#include <linux/fb.h>
> +#include <linux/init.h>
> +#include <linux/io.h>
> +#include <linux/platform_device.h>
> +#include <linux/of.h>
> +
> +/* OCFB register defines */
> +#define OCFB_CTRL	0x000
> +#define OCFB_STAT	0x004
> +#define OCFB_HTIM	0x008
> +#define OCFB_VTIM	0x00c
> +#define OCFB_HVLEN	0x010
> +#define OCFB_VBARA	0x014
> +#define OCFB_PALETTE	0x800
> +
> +#define OCFB_CTRL_VEN	0x00000001 /* Video Enable */
> +#define OCFB_CTRL_HIE	0x00000002 /* HSync Interrupt Enable */
> +#define OCFB_CTRL_PC	0x00000800 /* 8-bit Pseudo Color Enable*/
> +#define OCFB_CTRL_CD8	0x00000000 /* Color Depth 8 */
> +#define OCFB_CTRL_CD16	0x00000200 /* Color Depth 16 */
> +#define OCFB_CTRL_CD24	0x00000400 /* Color Depth 24 */
> +#define OCFB_CTRL_CD32	0x00000600 /* Color Depth 32 */
> +#define OCFB_CTRL_VBL1	0x00000000 /* Burst Length 1 */
> +#define OCFB_CTRL_VBL2	0x00000080 /* Burst Length 2 */
> +#define OCFB_CTRL_VBL4	0x00000100 /* Burst Length 4 */
> +#define OCFB_CTRL_VBL8	0x00000180 /* Burst Length 8 */
> +
> +#define PALETTE_SIZE	256
> +
> +#define OCFB_NAME	"OC VGA/LCD"
> +
> +static char *mode_option;
> +
> +static const struct fb_videomode default_mode = {
> +	/* 640x480 @ 60 Hz, 31.5 kHz hsync */
> +	NULL, 60, 640, 480, 39721, 40, 24, 32, 11, 96, 2,
> +	0, FB_VMODE_NONINTERLACED
> +};
> +
> +struct ocfb_dev {
> +	struct fb_info info;
> +	/* Physical and virtual addresses of control regs */
> +	phys_addr_t regs_phys;
> +	int regs_phys_size;
> +	void __iomem *regs;
> +	/* flag indicating whether the regs are little endian accessed */
> +	int little_endian;
> +	/* Physical and virtual addresses of framebuffer */
> +	phys_addr_t fb_phys;
> +	void __iomem *fb_virt;
> +	u32 pseudo_palette[PALETTE_SIZE];
> +};
> +
> +struct ocfb_par {
> +	void __iomem *pal_adr;
> +};
> +
> +static struct ocfb_par ocfb_par_priv;
> +
> +static struct fb_var_screeninfo ocfb_var;
> +static struct fb_fix_screeninfo ocfb_fix;
> +
> +#ifndef MODULE
> +static int __init ocfb_setup(char *options)
> +{
> +	char *curr_opt;
> +
> +	if (!options || !*options)
> +		return 0;
> +
> +	while ((curr_opt = strsep(&options, ",")) != NULL) {
> +		if (!*curr_opt)
> +			continue;
> +		mode_option = curr_opt;
> +	}
> +
> +	return 0;
> +}
> +#endif
> +
> +static inline u32 ocfb_readreg(struct ocfb_dev *fbdev, loff_t offset)
> +{
> +	if (fbdev->little_endian)
> +		return ioread32(fbdev->regs + offset);
> +	else
> +		return ioread32be(fbdev->regs + offset);
> +}
> +
> +static void ocfb_writereg(struct ocfb_dev *fbdev, loff_t offset, u32 data)
> +{
> +	if (fbdev->little_endian)
> +		iowrite32(data, fbdev->regs + offset);
> +	else
> +		iowrite32be(data, fbdev->regs + offset);
> +}
> +
> +static int ocfb_setupfb(struct ocfb_dev *fbdev)
> +{
> +	unsigned long bpp_config;
> +	struct fb_var_screeninfo *var = &fbdev->info.var;
> +	struct device *dev = fbdev->info.device;
> +	u32 hlen;
> +	u32 vlen;
> +
> +	/* Disable display */
> +	ocfb_writereg(fbdev, OCFB_CTRL, 0);
> +
> +	/* Register framebuffer address */
> +	fbdev->little_endian = 0;
> +	ocfb_writereg(fbdev, OCFB_VBARA, fbdev->fb_phys);
> +
> +	/* Detect endianess */
> +	if (ocfb_readreg(fbdev, OCFB_VBARA) != fbdev->fb_phys) {
> +		fbdev->little_endian = 1;
> +		ocfb_writereg(fbdev, OCFB_VBARA, fbdev->fb_phys);
> +	}
> +
> +	/* Horizontal timings */
> +	ocfb_writereg(fbdev, OCFB_HTIM, (var->hsync_len - 1) << 24 |
> +		      (var->right_margin - 1) << 16 | (var->xres - 1));
> +
> +	/* Vertical timings */
> +	ocfb_writereg(fbdev, OCFB_VTIM, (var->vsync_len - 1) << 24 |
> +		      (var->lower_margin - 1) << 16 | (var->yres - 1));
> +
> +	/* Total length of frame */
> +	hlen = var->left_margin + var->right_margin + var->hsync_len +
> +		var->xres;
> +
> +	vlen = var->upper_margin + var->lower_margin + var->vsync_len +
> +		var->yres;
> +
> +	ocfb_writereg(fbdev, OCFB_HVLEN, (hlen - 1) << 16 | (vlen - 1));
> +
> +	bpp_config = OCFB_CTRL_CD8;
> +	switch (var->bits_per_pixel) {
> +	case 8:
> +		if (!var->grayscale)
> +			bpp_config |= OCFB_CTRL_PC;  /* enable palette */
> +		break;
> +
> +	case 16:
> +		bpp_config |= OCFB_CTRL_CD16;
> +		break;
> +
> +	case 24:
> +		bpp_config |= OCFB_CTRL_CD24;
> +		break;
> +
> +	case 32:
> +		bpp_config |= OCFB_CTRL_CD32;
> +		break;
> +
> +	default:
> +		dev_err(dev, "no bpp specified\n");
> +		break;
> +	}
> +
> +	/* maximum (8) VBL (video memory burst length) */
> +	bpp_config |= OCFB_CTRL_VBL8;
> +
> +	/* Enable output */
> +	ocfb_writereg(fbdev, OCFB_CTRL, (OCFB_CTRL_VEN | bpp_config));
> +
> +	return 0;
> +}
> +
> +static int ocfb_setcolreg(unsigned regno, unsigned red, unsigned green,
> +			  unsigned blue, unsigned transp,
> +			  struct fb_info *info)
> +{
> +	struct ocfb_par *par = (struct ocfb_par *)info->par;
> +	u32 color;
> +
> +	if (regno >= info->cmap.len) {
> +		dev_err(info->device, "regno >= cmap.len\n");
> +		return 1;
> +	}
> +
> +	if (info->var.grayscale) {
> +		/* grayscale = 0.30*R + 0.59*G + 0.11*B */
> +		red = green = blue = (red * 77 + green * 151 + blue * 28) >> 8;
> +	}
> +
> +	red >>= (16 - info->var.red.length);
> +	green >>= (16 - info->var.green.length);
> +	blue >>= (16 - info->var.blue.length);
> +	transp >>= (16 - info->var.transp.length);
> +
> +	if (info->var.bits_per_pixel == 8 && !info->var.grayscale) {
> +		regno <<= 2;
> +		color = (red << 16) | (green << 8) | blue;
> +		ocfb_writereg(par->pal_adr, regno, color);
> +	} else {
> +		((u32 *)(info->pseudo_palette))[regno] =
> +			(red << info->var.red.offset) |
> +			(green << info->var.green.offset) |
> +			(blue << info->var.blue.offset) |
> +			(transp << info->var.transp.offset);
> +	}
> +
> +	return 0;
> +}
> +
> +static int ocfb_init_fix(struct ocfb_dev *fbdev)
> +{
> +	struct fb_var_screeninfo *var = &fbdev->info.var;
> +	struct fb_fix_screeninfo *fix = &fbdev->info.fix;
> +
> +	strcpy(fix->id, OCFB_NAME);
> +
> +	fix->line_length = var->xres * var->bits_per_pixel/8;
> +	fix->smem_len = fix->line_length * var->yres;
> +	fix->type = FB_TYPE_PACKED_PIXELS;
> +
> +	if (var->bits_per_pixel == 8 && !var->grayscale)
> +		fix->visual = FB_VISUAL_PSEUDOCOLOR;
> +	else
> +		fix->visual = FB_VISUAL_TRUECOLOR;
> +
> +	return 0;
> +}
> +
> +static int ocfb_init_var(struct ocfb_dev *fbdev)
> +{
> +	struct fb_var_screeninfo *var = &fbdev->info.var;
> +
> +	var->accel_flags = FB_ACCEL_NONE;
> +	var->activate = FB_ACTIVATE_NOW;
> +	var->xres_virtual = var->xres;
> +	var->yres_virtual = var->yres;
> +
> +	switch (var->bits_per_pixel) {
> +	case 8:
> +		var->transp.offset = 0;
> +		var->transp.length = 0;
> +		var->red.offset = 0;
> +		var->red.length = 8;
> +		var->green.offset = 0;
> +		var->green.length = 8;
> +		var->blue.offset = 0;
> +		var->blue.length = 8;
> +		break;
> +
> +	case 16:
> +		var->transp.offset = 0;
> +		var->transp.length = 0;
> +		var->red.offset = 11;
> +		var->red.length = 5;
> +		var->green.offset = 5;
> +		var->green.length = 6;
> +		var->blue.offset = 0;
> +		var->blue.length  = 5;
> +		break;
> +
> +	case 24:
> +		var->transp.offset = 0;
> +		var->transp.length = 0;
> +		var->red.offset = 16;
> +		var->red.length = 8;
> +		var->green.offset = 8;
> +		var->green.length = 8;
> +		var->blue.offset = 0;
> +		var->blue.length = 8;
> +		break;
> +
> +	case 32:
> +		var->transp.offset = 24;
> +		var->transp.length = 8;
> +		var->red.offset = 16;
> +		var->red.length = 8;
> +		var->green.offset = 8;
> +		var->green.length = 8;
> +		var->blue.offset = 0;
> +		var->blue.length = 8;
> +		break;
> +	}
> +
> +	return 0;
> +}
> +
> +static struct fb_ops ocfb_ops = {
> +	.owner		= THIS_MODULE,
> +	.fb_setcolreg	= ocfb_setcolreg,
> +	.fb_fillrect	= cfb_fillrect,
> +	.fb_copyarea	= cfb_copyarea,
> +	.fb_imageblit	= cfb_imageblit,
> +};
> +
> +static int ocfb_probe(struct platform_device *pdev)
> +{
> +	int ret = 0;
> +	struct ocfb_dev *fbdev;
> +	struct ocfb_par *par = &ocfb_par_priv;
> +	struct resource *res;
> +	struct resource *mmio;
> +	int fbsize;
> +
> +	fbdev = devm_kzalloc(&pdev->dev, sizeof(*fbdev), GFP_KERNEL);
> +	if (!fbdev)
> +		return -ENOMEM;
> +
> +	platform_set_drvdata(pdev, fbdev);
> +
> +	fbdev->info.fbops = &ocfb_ops;
> +	fbdev->info.var = ocfb_var;
> +	fbdev->info.fix = ocfb_fix;
> +	fbdev->info.device = &pdev->dev;
> +	fbdev->info.par = par;
> +
> +	/* Video mode setup */
> +	if (!fb_find_mode(&fbdev->info.var, &fbdev->info, mode_option,
> +			  NULL, 0, &default_mode, 16)) {
> +		dev_err(&pdev->dev, "No valid video modes found\n");
> +		return -EINVAL;
> +	}
> +	ocfb_init_var(fbdev);
> +	ocfb_init_fix(fbdev);
> +
> +	/* Request I/O resource */
> +	res = platform_get_resource(pdev, IORESOURCE_MEM, 0);
> +	if (!res) {
> +		dev_err(&pdev->dev, "I/O resource request failed\n");
> +		return -ENXIO;
> +	}
> +	fbdev->regs_phys = res->start;
> +	fbdev->regs_phys_size = resource_size(res);
> +	mmio = devm_request_mem_region(&pdev->dev, res->start,
> +				       resource_size(res), res->name);
> +	if (!mmio) {
> +		dev_err(&pdev->dev, "I/O memory space request failed\n");
> +		return -ENXIO;
> +	}
> +	fbdev->regs = devm_ioremap_nocache(&pdev->dev, mmio->start,
> +					   resource_size(mmio));
> +	if (!fbdev->regs) {
> +		dev_err(&pdev->dev, "I/O memory remap request failed\n");
> +		return -ENXIO;
> +	}


This construction seems to me too complicated and can be simpler.

What about this?
	res = platform_get_resource(pdev, IORESOURCE_MEM, 0);
	res->flags &= ~IORESOURCE_CACHEABLE;
	fbdev->regs = devm_ioremap_resource(&pdev->dev, res);
	if (IS_ERR(fbdev->regs))
		return PTR_ERR(fbdev->regs);

Thanks,
Michal


-- 
Michal Simek, Ing. (M.Eng), OpenPGP -> KeyID: FE3D1F91
w: www.monstr.eu p: +42-0-721842854
Maintainer of Linux kernel - Microblaze cpu - http://www.monstr.eu/fdt/
Maintainer of Linux kernel - Xilinx Zynq ARM architecture
Microblaze U-BOOT custodian and responsible for u-boot arm zynq platform



[-- Attachment #2: OpenPGP digital signature --]
[-- Type: application/pgp-signature, Size: 263 bytes --]

  parent reply	other threads:[~2013-11-26  7:10 UTC|newest]

Thread overview: 14+ messages / expand[flat|nested]  mbox.gz  Atom feed  top
2013-11-22  4:34 [PATCH v3] video: add OpenCores VGA/LCD framebuffer driver Stefan Kristiansson
2013-11-22  4:34 ` Stefan Kristiansson
2013-11-24 14:12 ` Jean-Christophe PLAGNIOL-VILLARD
2013-11-24 14:12   ` Jean-Christophe PLAGNIOL-VILLARD
2013-11-26  6:55   ` Stefan Kristiansson
2013-11-26  6:55     ` Stefan Kristiansson
2013-11-26  7:10 ` Michal Simek [this message]
2013-11-26  7:10   ` Michal Simek
2013-11-26  8:13   ` Stefan Kristiansson
2013-11-26  8:13     ` Stefan Kristiansson
2013-11-26  8:19   ` Geert Uytterhoeven
2013-11-26  8:19     ` Geert Uytterhoeven
2013-11-26  8:22     ` Michal Simek
2013-11-26  8:22       ` Michal Simek

Reply instructions:

You may reply publicly to this message via plain-text email
using any one of the following methods:

* Save the following mbox file, import it into your mail client,
  and reply-to-all from there: mbox

  Avoid top-posting and favor interleaved quoting:
  https://en.wikipedia.org/wiki/Posting_style#Interleaved_style

* Reply using the --to, --cc, and --in-reply-to
  switches of git-send-email(1):

  git send-email \
    --in-reply-to=52944976.4060003@monstr.eu \
    --to=monstr@monstr.eu \
    --cc=linux-fbdev@vger.kernel.org \
    --cc=linux-kernel@vger.kernel.org \
    --cc=plagnioj@jcrosoft.com \
    --cc=stefan.kristiansson@saunalahti.fi \
    --cc=tomi.valkeinen@ti.com \
    /path/to/YOUR_REPLY

  https://kernel.org/pub/software/scm/git/docs/git-send-email.html

* If your mail client supports setting the In-Reply-To header
  via mailto: links, try the mailto: link
Be sure your reply has a Subject: header at the top and a blank line before the message body.
This is an external index of several public inboxes,
see mirroring instructions on how to clone and mirror
all data and code used by this external index.