上篇博客中进行了lcd的简单测试,这篇博客将进行更加复杂的测试——画点、画线、画圆。画线和画圆是在画点的基础上实现的,因此本篇博客重点实现画点操作。

先抛出这样的一个问题,已知:

(x,y)的坐标;

bpp;

xres;

yres;

那么,如何在framebuffer中获得像素的地址呢?

(x,y)像素的起始地址 = fb_base +(xres * bpp /8)* y +x * bpp/8

(xres * bpp /8)表示一行占据多少个字节,乘以y表示y行共占据多少个像素

2)在framebuffer.c中实现画点操作

 #include "lcd.h"

 /* 实现画点 */

 /* 获得LCD参数 */
static unsigned int fb_base;
static int xres, yres, bpp; void fb_get_lcd_params(void)
{
get_lcd_params(&fb_base, &xres, &yres, &bpp);
} /* rgb: 0x00RRGGBB */
unsigned short convert32bppto16bpp(unsigned int rgb)
{
int r = (rgb >> )& 0xff; /*将红色的值拿到*/
int g = (rgb >> ) & 0xff; /*将绿色的值拿到*/
int b = rgb & 0xff; /*将蓝色的值拿到*/ /* rgb565 */ /*需要将rgb组成一个16位的数,怎么组*/
r = r >> ; /*将红色的值右移3位,相当于取高5位。为什么这么取,得看16bpp数据的组织格式*/
g = g >> ; /*将绿色的值右移2位,相当于取高6位。为什么这么取,得看16bpp数据的组织格式*/
b = b >> ; /*将蓝色的值右移3位,相当于取高5位。为什么这么取,得看16bpp数据的组织格式*/ return ((r<<) | (g<<) | (b)); /*组合成16位的数据*/
} /* color : 32bit, 0x00RRGGBB
* color是32位的,用的lcd是16位的,因此需要将color转换成是16位的。
*/
void fb_put_pixel(int x, int y, unsigned int color)
{
unsigned char *pc; /* 8bpp */
unsigned short *pw; /* 16bpp */
unsigned int *pdw; /* 32bpp */ unsigned int pixel_base = fb_base + (xres * bpp / ) * y + x * bpp / ; switch (bpp)
{
case :
pc = (unsigned char *) pixel_base;
*pc = color; /*这个地方是有问题的,之所以将color直接赋给*pc,是因为程序压根不会走到这。因为所使用的lcd是16位的。*/
break;
case :
pw = (unsigned short *) pixel_base;
*pw = convert32bppto16bpp(color);
break;
case :
pdw = (unsigned int *) pixel_base;
*pdw = color;
break;
}
}

画线、画圆的操作在geometry.c中实现,它不是我们的重点,直接拿别人的使用即可。只需要将自己写的画点的函数fb_put_pixel用上即可

 #include "framebuffer.h"

 /*
* http://blog.csdn.net/p1126500468/article/details/50428613
*/ //-------------画圆函数。参数:圆心,半径,颜色----------
// 画1/8圆 然后其他7/8对称画
// ---------------->X
// |(0,0) 0
// | 7 1
// | 6 2
// | 5 3
// (Y)V 4
//
// L = x^2 + y^2 - r^2
void draw_circle(int x, int y, int r, int color)
{
int a, b, num;
a = ;
b = r;
while( * b * b >= r * r) // 1/8圆即可
{
fb_put_pixel(x + a, y - b,color); // 0~1
fb_put_pixel(x - a, y - b,color); // 0~7
fb_put_pixel(x - a, y + b,color); // 4~5
fb_put_pixel(x + a, y + b,color); // 4~3 fb_put_pixel(x + b, y + a,color); // 2~3
fb_put_pixel(x + b, y - a,color); // 2~1
fb_put_pixel(x - b, y - a,color); // 6~7
fb_put_pixel(x - b, y + a,color); // 6~5 a++;
num = (a * a + b * b) - r*r;
if(num > )
{
b--;
a--;
}
}
} //-----------画线。参数:起始坐标,终点坐标,颜色--------
void draw_line(int x1,int y1,int x2,int y2,int color)
{
int dx,dy,e;
dx=x2-x1;
dy=y2-y1;
if(dx>=)
{
if(dy >= ) // dy>=0
{
if(dx>=dy) // 1/8 octant
{
e=dy-dx/;
while(x1<=x2)
{
fb_put_pixel(x1,y1,color);
if(e>){y1+=;e-=dx;}
x1+=;
e+=dy;
}
}
else // 2/8 octant
{
e=dx-dy/;
while(y1<=y2)
{
fb_put_pixel(x1,y1,color);
if(e>){x1+=;e-=dy;}
y1+=;
e+=dx;
}
}
}
else // dy<0
{
dy=-dy; // dy=abs(dy)
if(dx>=dy) // 8/8 octant
{
e=dy-dx/;
while(x1<=x2)
{
fb_put_pixel(x1,y1,color);
if(e>){y1-=;e-=dx;}
x1+=;
e+=dy;
}
}
else // 7/8 octant
{
e=dx-dy/;
while(y1>=y2)
{
fb_put_pixel(x1,y1,color);
if(e>){x1+=;e-=dy;}
y1-=;
e+=dx;
}
}
}
}
else //dx<0
{
dx=-dx; //dx=abs(dx)
if(dy >= ) // dy>=0
{
if(dx>=dy) // 4/8 octant
{
e=dy-dx/;
while(x1>=x2)
{
fb_put_pixel(x1,y1,color);
if(e>){y1+=;e-=dx;}
x1-=;
e+=dy;
}
}
else // 3/8 octant
{
e=dx-dy/;
while(y1<=y2)
{
fb_put_pixel(x1,y1,color);
if(e>){x1-=;e-=dy;}
y1+=;
e+=dx;
}
}
}
else // dy<0
{
dy=-dy; // dy=abs(dy)
if(dx>=dy) // 5/8 octant
{
e=dy-dx/;
while(x1>=x2)
{
fb_put_pixel(x1,y1,color);
if(e>){y1-=;e-=dx;}
x1-=;
e+=dy;
}
}
else // 6/8 octant
{
e=dx-dy/;
while(y1>=y2)
{
fb_put_pixel(x1,y1,color);
if(e>){x1-=;e-=dy;}
y1-=;
e+=dx;
}
}
}
}
}

3)在led_test.c中,需要测什么?看下图:

 #include "geometry.h"

 void lcd_test(void)
{
unsigned int fb_base;
int xres, yres, bpp;
int x, y;
unsigned short *p;
unsigned int *p2; /* 初始化LCD */
lcd_init(); /* 使能LCD */
lcd_enable(); /* 获得LCD的参数: fb_base, xres, yres, bpp */
get_lcd_params(&fb_base, &xres, &yres, &bpp);
fb_get_lcd_params(); /* 往framebuffer中写数据 */
if (bpp == )
{
/* 让LCD输出整屏的红色 */ /* 565: 0xf800 */ p = (unsigned short *)fb_base;
for (x = ; x < xres; x++)
for (y = ; y < yres; y++)
*p++ = 0xf800; /* green */
p = (unsigned short *)fb_base;
for (x = ; x < xres; x++)
for (y = ; y < yres; y++)
*p++ = 0x7e0; /* blue */
p = (unsigned short *)fb_base;
for (x = ; x < xres; x++)
for (y = ; y < yres; y++)
*p++ = 0x1f; /* black */
p = (unsigned short *)fb_base;
for (x = ; x < xres; x++)
for (y = ; y < yres; y++)
*p++ = ; } delay(); /* 画线 */
draw_line(, , xres - , , 0xff0000);
draw_line(xres - , , xres - , yres - , 0xffff00);
draw_line(, yres - , xres - , yres - , 0xff00aa);
draw_line(, , , yres - , 0xff00ef);
draw_line(, , xres - , yres - , 0xff4500);
draw_line(xres - , , , yres - , 0xff0780); delay(); /* 画圆 */
draw_circle(xres/, yres/, yres/, 0xff00);
}

所看到的现象就是,lcd显输出整屏红色----->绿色------->蓝色------->黑色---------->画线--------->画圆

最新文章

  1. AngularJs最简单解决跨域问题案例
  2. 【原】iOS学习之PCH文件
  3. JavaIO中的Reader和writer
  4. Quartz.net开源作业调度框架使用详解(转)
  5. android LayoutInflater.inflate()的参数介绍
  6. Why longest path problem doesn&#39;t have optimal substructure?
  7. GCC内联汇编入门
  8. C++から広がり
  9. RTMP直播应用与延时分析
  10. 创建 userSettings/Microsoft.SqlServer.Configuration.LandingPage.Properties.Settings 的配置节处理程序时出错: 未能加载文件或程序集“System, Version=4.0.0.0, Culture=neutral, PublicKeyToken=b77a5c561934e089”或它的某一个依赖项。系统没找到指定的文件
  11. STAR法则的感想
  12. 暑假练习赛 007 E - Pairs
  13. C#学习笔记-基础知识篇(不定期更新)
  14. jmeter接口/性能压测入门
  15. Python 正则表达式相关问题
  16. 解决Kubelet Pod启动CreatePodSandbox或RunPodSandbox异常方法
  17. docker执行第一个应用
  18. 二维数组转化为一维数组 contact 与apply 的结合
  19. GridView中CheckBox单击事件(oncheckedchanged)
  20. atom介绍

热门文章

  1. 状压DP(超详细!!!)
  2. Computer-Hunters——冲刺总结
  3. [数据库] SQL 语法之进阶篇
  4. Java8 新特性 Stream() 创建流
  5. SQL --------------- 运算符 = 与 in
  6. jmeter 如何获取一小时之前的时间戳
  7. Git更改文件名大小写,提交失败问题解决
  8. java设计模式单例模式
  9. windows系统mysql-5.7官方绿色版zip包安装教程
  10. Layui 时间转换时间戳