linux键盘驱动程序分析,基于Linux按键驱动分析与编程
硬件平台:Mini2440
Size of NAND:256M
linux kernel:linux-2.6.32.2
一、首先编写按键驱动要用到的Mini2440的硬件是中断控制器和定时器
那么linux-2.6.32.2的中断号的预定义文件IRQ(X)是在2.6.32.2/arch/arm/mach-s3c2410/include/mach的irq.h的头文件中如下:
#ifndef __ASM_ARCH_IRQS_H
#define __ASM_ARCH_IRQS_H __FILE__
#define S3C2410_CPUIRQ_OFFSET (16)
#define S3C2410_IRQ(x) ((x) + S3C2410_CPUIRQ_OFFSET)
#define
IRQ_EINT0 S3C2410_IRQ(0) #define
IRQ_EINT1 S3C2410_IRQ(1)
#define
IRQ_EINT2 S3C2410_IRQ(2)
#define
IRQ_EINT3 S3C2410_IRQ(3)
#define
IRQ_EINT4t7 S3C2410_IRQ(4) #define IRQ_EINT8t23 S3C2410_IRQ(5)
#define IRQ_RESERVED6 S3C2410_IRQ(6) #define
IRQ_CAM S3C2410_IRQ(6) #define IRQ_BATT_FLT S3C2410_IRQ(7)
#define
IRQ_TICK S3C2410_IRQ(8) #define
IRQ_WDT S3C2410_IRQ(9) #define
IRQ_TIMER0 S3C2410_IRQ(10)
#define
IRQ_TIMER1 S3C2410_IRQ(11)
#define
IRQ_TIMER2 S3C2410_IRQ(12)
#define
IRQ_TIMER3 S3C2410_IRQ(13)
#define
IRQ_TIMER4 S3C2410_IRQ(14)
#define
IRQ_UART2 S3C2410_IRQ(15)
#define
IRQ_LCD S3C2410_IRQ(16) #define
IRQ_DMA0 S3C2410_IRQ(17) #define
IRQ_DMA1 S3C2410_IRQ(18)
#define
IRQ_DMA2 S3C2410_IRQ(19)
#define
IRQ_DMA3 S3C2410_IRQ(20)
#define
IRQ_SDI S3C2410_IRQ(21)
#define
IRQ_SPI0 S3C2410_IRQ(22)
#define
IRQ_UART1 S3C2410_IRQ(23)
#define IRQ_RESERVED24
S3C2410_IRQ(24) #define
IRQ_NFCON S3C2410_IRQ(24) #define
IRQ_USBD S3C2410_IRQ(25)
#define
IRQ_USBH S3C2410_IRQ(26)
#define
IRQ_IIC S3C2410_IRQ(27)
#define
IRQ_UART0 S3C2410_IRQ(28) #define
IRQ_SPI1 S3C2410_IRQ(29)
#define
IRQ_RTC S3C2410_IRQ(30)
#define IRQ_ADCPARENT S3C2410_IRQ(31)
#define
IRQ_EINT4 S3C2410_IRQ(32) #define
IRQ_EINT5 S3C2410_IRQ(33)
#define
IRQ_EINT6 S3C2410_IRQ(34)
#define
IRQ_EINT7 S3C2410_IRQ(35)
#define
IRQ_EINT8 S3C2410_IRQ(36)
#define
IRQ_EINT9 S3C2410_IRQ(37)
#define
IRQ_EINT10 S3C2410_IRQ(38)
#define
IRQ_EINT11 S3C2410_IRQ(39)
#define
IRQ_EINT12 S3C2410_IRQ(40)
#define
IRQ_EINT13 S3C2410_IRQ(41)
#define
IRQ_EINT14 S3C2410_IRQ(42)
#define
IRQ_EINT15 S3C2410_IRQ(43)
#define
IRQ_EINT16 S3C2410_IRQ(44)
#define
IRQ_EINT17 S3C2410_IRQ(45)
#define
IRQ_EINT18 S3C2410_IRQ(46)
#define
IRQ_EINT19 S3C2410_IRQ(47)
#define
IRQ_EINT20 S3C2410_IRQ(48) #define
IRQ_EINT21 S3C2410_IRQ(49)
#define
IRQ_EINT22 S3C2410_IRQ(50)
#define
IRQ_EINT23 S3C2410_IRQ(51)
#define IRQ_EINT_BIT(x) ((x) - IRQ_EINT4 +
4)
#define
IRQ_EINT(x) (((x) >= 4) ? (IRQ_EINT4 + (x) - 4) : (IRQ_EINT0 + (x)))
#define IRQ_LCD_FIFO S3C2410_IRQ(52)
#define IRQ_LCD_FRAME S3C2410_IRQ(53)
#define
S3C2410_IRQSUB(x) S3C2410_IRQ((x)+54)
#define
IRQ_S3CUART_RX0 S3C2410_IRQSUB(0) #define
IRQ_S3CUART_TX0 S3C2410_IRQSUB(1)
#define IRQ_S3CUART_ERR0 S3C2410_IRQSUB(2)
#define
IRQ_S3CUART_RX1 S3C2410_IRQSUB(3) #define
IRQ_S3CUART_TX1 S3C2410_IRQSUB(4)
#define IRQ_S3CUART_ERR1 S3C2410_IRQSUB(5)
#define
IRQ_S3CUART_RX2 S3C2410_IRQSUB(6) #define
IRQ_S3CUART_TX2 S3C2410_IRQSUB(7)
#define IRQ_S3CUART_ERR2 S3C2410_IRQSUB(8)
#define
IRQ_TC S3C2410_IRQSUB(9)
#define
IRQ_ADC S3C2410_IRQSUB(10)
#define IRQ_S3C2412_CFSDI S3C2410_IRQ(21)
#define
IRQ_S3C2412_SDI S3C2410_IRQSUB(13)
#define
IRQ_S3C2412_CF S3C2410_IRQSUB(14)
#define
IRQ_S3C2440_CAM_C S3C2410_IRQSUB(11) #define
IRQ_S3C2440_CAM_P S3C2410_IRQSUB(12) #define
IRQ_S3C2440_WDT S3C2410_IRQSUB(13)
#define IRQ_S3C2440_AC97 S3C2410_IRQSUB(14)
#define
IRQ_S3C2443_DMA S3C2410_IRQ(17) #define
IRQ_S3C2443_UART3 S3C2410_IRQ(18) #define
IRQ_S3C2443_CFCON S3C2410_IRQ(19) #define
IRQ_S3C2443_HSMMC S3C2410_IRQ(20) #define
IRQ_S3C2443_NAND S3C2410_IRQ(24)
#define
IRQ_HSMMC0 IRQ_S3C2443_HSMMC
#define
IRQ_S3C2443_LCD1 S3C2410_IRQSUB(14)
#define IRQ_S3C2443_LCD2 S3C2410_IRQSUB(15)
#define IRQ_S3C2443_LCD3 S3C2410_IRQSUB(16)
#define IRQ_S3C2443_LCD4 S3C2410_IRQSUB(17)
#define
IRQ_S3C2443_DMA0 S3C2410_IRQSUB(18)
#define IRQ_S3C2443_DMA1 S3C2410_IRQSUB(19)
#define IRQ_S3C2443_DMA2 S3C2410_IRQSUB(20)
#define IRQ_S3C2443_DMA3 S3C2410_IRQSUB(21)
#define IRQ_S3C2443_DMA4 S3C2410_IRQSUB(22)
#define IRQ_S3C2443_DMA5 S3C2410_IRQSUB(23)
#define
IRQ_S3C2443_RX3 S3C2410_IRQSUB(24)
#define
IRQ_S3C2443_TX3 S3C2410_IRQSUB(25)
#define IRQ_S3C2443_ERR3 S3C2410_IRQSUB(26)
#define
IRQ_S3C2443_WDT S3C2410_IRQSUB(27)
#define IRQ_S3C2443_AC97 S3C2410_IRQSUB(28)
#ifdef CONFIG_CPU_S3C2443
#define NR_IRQS (IRQ_S3C2443_AC97+1)
#else
#define NR_IRQS (IRQ_S3C2440_AC97+1)
#endif
#define
IRQ_UART3 IRQ_S3C2443_UART3
#define
IRQ_S3CUART_RX3 IRQ_S3C2443_RX3
#define
IRQ_S3CUART_TX3 IRQ_S3C2443_TX3
#define IRQ_S3CUART_ERR3 IRQ_S3C2443_ERR3
#ifdef CONFIG_CPU_S3C2440
#define IRQ_S3C244x_AC97 IRQ_S3C2440_AC97
#else
#define IRQ_S3C244x_AC97 IRQ_S3C2443_AC97
#endif
#define FIQ_START IRQ_EINT0
#endif
//
那么linux下面的定义的中断函数在Linux-2.6.32.2/kernel/irq/manage.c里面。
中断的预定义处理命令函数则定义在Linux-2.6.32.2/include/linux/irq.h和interrupt.h里面。
二、驱动使用到中断之外,按键的驱动程序还需要用到定时器。那么定时器在按键驱动程序这边的作用是去除按键的抖动作用,一般来说当按键驱动程序检测到按键按下之后,还要启动Linux内部的定时器延时20ms左右,然后再去检查相应的引脚是否还是低电平。若还是低电平则表示按键是真的有按下,若这时候检测到引脚是高电平那么则可以判断按键比没有真正的按下去,而是有可能是外部的干扰所致的。所以按键驱动程序中的定时器对于一个稳定可靠的按键驱动程序时至关重要的。
1、Linux的内核定时器是内核用来控制在未来某个时间点(基于jiffies)调度执行某个函数的一种机制,其实现位于 和 kernel/timer.c 文件中。
内核定时器是内核用来控制在未来某个时间点(基于jiffies)调度执行某个函数的一种机制,其实现位于和
kernel/timer.c 文件中。
2、按键驱动的流程图
三、驱动程序
#include
#include
#include
#include
#include
#include
#include
#include
#include
#include
#include
#include
#include
#include
#include
#include
#include
#include
#include
#define uchar unsigned char
#define uint unsigned
int
#define CLEAR_CACHE 0x0f
#define KEY_STATUES_X
0 //按键未确定的标识
#define
KEY_DOWN 1 //按键按下去的标识
#define
KEY_UP 2 //按键释放的标识
#define KEY_MAJOR 250
//按键驱动程序的主设备号(主设备号不是固定的)
#define KEY_SUBMAJOR
0 //按键驱动程序的次设备号
#define DEVICE_NAME "key_drive"
//按键驱动程序的设备名
#define KEY_NUM
6 //按键的数量
#define NUM_KEY_STATE KEY_NUM //按键的状态数量
#define NUM_KEY_CACHE
16 //按键的缓冲器
#define MOD
NUM_KEY_CACHE //按键的循环指针
#define DELAY_TIME_20MS
HZ/50 //按键延时
#define KEY_STATUES_20MS DELAY_TIME_20MS
#define DELAY_TIME_100MS HZ/10
#define KEY_STATUES_100MS DELAY_TIME_100MS
#define POINTER_BUFFER_CYCLE(x) ((++x)&(MOD-1))
//按键的缓冲循环指针定义
static void hander_timer(unsigned long key_num);
//按键的定时器
void key_event(void); //按键的时间产生函数
static void key_setup_cdev(void);
static volatile unsigned char interrupt_num;
unsigned int key_major=KEY_MAJOR;
unsigned long flags;
static struct file_operations key_operation;
struct key_char_dev
{
struct
cdev cdev;
uchar
key[NUM_KEY_STATE];
uchar
key_cache[NUM_KEY_CACHE]; //按键的缓冲器
uint head_read; //按键的读缓冲指针
uint head_writer; //按键的写缓冲指针
struct
timer_list key_timer[KEY_NUM]; //定时器结构体
wait_queue_head_t
key_queue; //等待队列头
};
struct key_info
{
uint
irq_num; //按键对应的中断号
uint
gpio_num; //按键对应的引脚
uchar
key_num; //按键本身的标号
};
struct key_char_dev *key_cdev;
struct key_info key_inform[KEY_NUM]={
{IRQ_EINT8 ,S3C2410_GPG(0), 0},
{IRQ_EINT11,S3C2410_GPG(3), 1},
{IRQ_EINT13,S3C2410_GPG(5), 2},
{IRQ_EINT14,S3C2410_GPG(6), 3},
{IRQ_EINT15,S3C2410_GPG(7), 4},
{IRQ_EINT19,S3C2410_GPG(11),5}
};
static unsigned int key_down(unsigned int key)
{
return s3c2410_gpio_getpin(key);
}
static void init_key_timer(void)
{
unsigned
char i;
for(i=0;i
{
init_timer(&(key_cdev->key_timer[i]));
key_cdev->key_timer[i].data=(unsigned long)i;
key_cdev->key_timer[i].function=hander_timer;
}
}
static void hander_timer(unsigned long key_num)
{
//先判断对应引脚的电平是否为低电平
if(key_down((unsigned int)(key_inform[key_num].gpio_num))==0)
{
//如果按键上一次的状态是不确定的状态,那么再一次检测到引脚的电平是低电平。那么这时就可以确定按键是按下去,则设置对应按键标识缓冲器为“按下去”的状态。
if(key_cdev->key[key_num]==KEY_STATUES_X)
{
key_cdev->key[key_num]=KEY_DOWN;
key_event(); //这时候启动定时器进行延时
key_cdev->key_timer[key_num].expires=jiffies+KEY_STATUES_20MS;
add_timer(&(key_cdev->key_timer[key_num]));
printk("\n Go in the here:judge_function: irq %d",key_num);
}
else
{
//如果按键的状态是已经处于按下去的状态,并且在一次检测到按键还没弹起来,则继续启动定时器的延时,知道按键被释放为止。
key_cdev->key_timer[key_num].expires=jiffies+KEY_STATUES_100MS;
add_timer(&(key_cdev->key_timer[key_num]));
}
}
else
{
//检测到按键被释放,则设置按键的状态为释放状态,并且重新使能该按键的中断。
key_cdev->key[key_num]=KEY_UP;
enable_irq(key_inform[key_num].irq_num);
printk("\n Go in the here:hander_timer: irq %d",key_num);
}
}
//按键的中断处理函数
static irqreturn_t key_hander(int irq,void
*dev_id) //The type of the irqreturn_t is
'int';
{
unsigned
char i;
unsigned
int get_irq;
get_irq=irq;
printk("\n Go in the here 1: irq");
//查找是由哪一个按键产生触发中断的事件。
for(i=0;i
{
if(key_inform[i].irq_num==get_irq)
{
interrupt_num=i;
break;
}
}
printk("\n Go in the here 2: irq");
key_cdev->key[interrupt_num]=KEY_STATUES_X;
//当检测到有相应的按键按下去的时候,则需要关闭对应的按键中断标识位,以防止发生再次的中断。
disable_irq_nosync(key_inform[interrupt_num].irq_num);
//设置定时器定时20ms,检查是否按键有按下
key_cdev->key_timer[interrupt_num].expires=jiffies+DELAY_TIME_20MS;
printk("\n Go in the here 3: irq %d",interrupt_num);
add_timer(&key_cdev->key_timer[interrupt_num]);
printk("\n Go in the here 4: irq");
return
interrupt_num;
}
//按键事件的处理函数
void key_event(void)
{
//增加按键的写头指针
key_cdev->head_writer=POINTER_BUFFER_CYCLE(key_cdev->head_writer);
//把按键号存入到按键的缓冲器中,以便应用程序能够读取
key_cdev->key_cache[key_cdev->head_writer]=interrupt_num;
//唤醒按键的读程序
wake_up_interruptible(&key_cdev->key_queue);
printk("\n Go in the here:head_writer
%d",key_cdev->head_writer);
printk("\n Go in the here:head_read
%d",key_cdev->head_read);
printk("\n Go in the here:key_event: irq %d",interrupt_num);
}
//加载字符设备
static void key_setup_cdev(void)
{
int
ret_num;
dev_t
dev_num;
dev_num=MKDEV(key_major,KEY_SUBMAJOR);
cdev_init(&key_cdev->cdev,&key_operation);
key_cdev->cdev.owner=THIS_MODULE;
ret_num=cdev_add(&key_cdev->cdev,dev_num,1);
if(ret_num<0)
{
printk("Error adding dev ");
}
}
static int __init key_dev_init(void)
{
//注册字符设备
dev_t
dev_num;
int
ret_num,i,error_irq;
dev_num=MKDEV(key_major,KEY_SUBMAJOR);
if(dev_num)
{
ret_num=register_chrdev_region(dev_num,1,"key_drive");
}
else
{
ret_num=alloc_chrdev_region(&dev_num,0,1,"key_drive");
}
if(ret_num<0)
{
return ret_num;
}
key_cdev=kmalloc(sizeof(struct key_char_dev),GFP_KERNEL);
if(!key_cdev)
{
ret_num=-ENOMEM;
goto fail_request;
}
memset(key_cdev,0,sizeof(struct key_char_dev));
key_setup_cdev();
//初始化中断
for(i=0;i
{
set_irq_type(key_inform[i].irq_num,IRQF_TRIGGER_FALLING);
error_irq=request_irq(key_inform[i].irq_num,key_hander,IRQF_DISABLED,"key_drive",NULL);
if(error_irq<0)
{
if(error_irq==-EINVAL)
{
printk("error:The interrupt number is invalid");
}
else if(error_irq==-EBUSY)
{
printk("error:The interrupt number have been employed");
}
}
}
//初始化等待队列头
init_waitqueue_head(&(key_cdev->key_queue));//等待队列头文件的初始化
init_key_timer();//定时器的初始化
fail_request:
unregister_chrdev_region(&dev_num,1);
return ret_num;
}
static int key_open(struct inode
*inodep,struct file *filep)
{
printk("succeful key open");
key_cdev->head_read=key_cdev->head_writer=0;
return
0;
}
static ssize_t key_read(struct file *filp,char __user
*buf,size_t count,loff_t *pps)
{
unsigned
char keynum;
unsigned
long error;
re_check:
if(key_cdev->head_read!=key_cdev->head_writer)
{
local_irq_save(flags);
keynum=key_cdev->key_cache[key_cdev->head_read];
key_cdev->head_read=POINTER_BUFFER_CYCLE(key_cdev->head_read);
error=copy_to_user(buf,&keynum,sizeof(unsigned char));
local_irq_restore(flags);
if(error)
{
printk("error:The copy_to_user is invalid");
return 0;
}
return sizeof(unsigned
char);
}
else
{
if(filp->f_flags&O_NONBLOCK) //在应用程序打开时如果OPEN函数有指定O_NONBLOCK参数那么将在这里起作用
{ //O_NONBLOCK这个参数就是驱动程序读数据时是执行非阻塞操
return
-EAGAIN; //This is a once again flag
}
//如果不能立即获取按键的值,或者根本就没有按键按下去,则按键驱动程序进入睡眠状态
interruptible_sleep_on(&(key_cdev->key_queue));
goto re_check;
}
return 0;
}
static int key_ioctl(struct inode *inodep,struct file
*filp,unsigned int cmd)
{
unsigned
char i;
switch(cmd)
{
case CLEAR_CACHE :
local_irq_disable();
key_cdev->head_writer=key_cdev->head_read=0;
for(i=0;i
{
key_cdev->key_cache[NUM_KEY_CACHE]=0;
}
local_irq_enable();
break;
default:
return -EINVAL;
}
return 0;
}
static struct file_operations key_operation={
.open=key_open,
.read=key_read,
.ioctl=key_ioctl,
.owner=THIS_MODULE
};
static void __exit key_dev_exit(void)
{
dev_t
dev_num;
unsigned
char i;
dev_num=MKDEV(key_major,KEY_SUBMAJOR);
unregister_chrdev_region(dev_num,1);
cdev_del(&(key_cdev->cdev));
kfree(key_cdev);
key_cdev=NULL;
for(i=0;i
{
free_irq(key_inform[i].irq_num,NULL);
}
}
module_init(key_dev_init);
module_exit(key_dev_exit);
Makefile文件:
1 obj-m := key.o
2 CC :=arm-linux-gcc
3 KDIR := /opt/linux-2.6.32.2
4 PWD := $(shell pwd)
5 default:
6 $(MAKE) -C $(KDIR) SUBDIRS=$(PWD) modules
7 clean:
8 rm -rf *.o
9 rm -rf *.ko
10 rm -rf .*.cmd
11 rm -rf *.mod.*
12
三、测试程序
#include
#include
#include
#include
#include
#include
int main(void)
{
int fd,come_value;
unsigned char Buff;
fd=open("/dev/key",O_NONBLOCK);
if(fd<0)
{
printf("error: Open the file come out error");
exit(1);
}
while(1)
{
read(fd,&Buff,sizeof(unsigned char));
printf("The keys value is :%d\n",Buff);
// sleep(1);
}
close(fd);
return
0;
}
Makefile文件:
1 CC := arm-linux-gcc
2
3 key_app : key_app.o
4 $(CC) -o key_app key_app.c
5
6 clean:
7 rm -rf key_*.o*~
8
linux键盘驱动程序分析,基于Linux按键驱动分析与编程相关推荐
- Linux keypad 设备树,matrix_keypad 矩阵按键驱动分析
matrix_keypad 矩阵按键驱动分析 //主要函数调用过程 matrix_keypad_probe matrix_keypad_parse_dt //根据设备树构造 pdata pdata-& ...
- 【linux驱动分析】之dm9000驱动分析
[linux驱动分析]之dm9000驱动分析(一):dm9000原理及硬件分析 [linux驱动分析]之dm9000驱动分析(一):dm9000原理及硬件分析 [linux驱动分析]之dm9000驱动 ...
- 深入学习Linux摄像头(三)虚拟摄像头驱动分析
深入学习Linux摄像头系列 深入学习Linux摄像头(一)v4l2应用编程 深入学习Linux摄像头(二)v4l2驱动框架 深入学习Linux摄像头(三)虚拟摄像头驱动分析 深入学习Linux摄像头 ...
- 【linux驱动分析】之dm9000驱动分析(三):sk_buff结构分析
[linux驱动分析]之dm9000驱动分析(一):dm9000原理及硬件分析 [linux驱动分析]之dm9000驱动分析(二):定义在板文件里的资源和设备以及几个宏 [linux驱动分析]之dm9 ...
- linux 嵌入式 快照_基于Linux的嵌入式启动优化的研究
基于Linux的嵌入式启动优化的研究 论文 开题分析 周报 启动 基于Linux嵌入式系统启动加速的研究 一. 所选课题的目的和意义 随着科学技术的蓬勃发展,计算机被更多的人使用,并与人类对额生活结合 ...
- linux dhcp 论文,毕业论文—基于linux的dhcp服务器配置.doc
毕业论文-基于linux的dhcp服务器配置 基于Linux的DHCP服务器配置 摘 要 随着互联网的普及,人们的工作.学习和生活与网络联系越来越紧密,搭建了许多不同的网络,如企业网.校园网和城区网等 ...
- linux无线网卡驱动分析,基于Linux的无线网卡驱动程序
基于Linux的无线网卡驱动程序 文章分析了Linux下无线网卡驱动程序结构及设计方法,着重 (本文共4页) 阅读全文>> 无线局域网是当前较为常见的一种无线接入技术,具有运行速度高一级灵 ...
- linux 网卡驱动分析,基于linux下网卡驱动分析及实现技术研究
摘 要 Linux技术是当前计算机技术中最大的一个热点,在我国以及全世界得到了迅猛的发展,被广泛的应用于嵌入式系统.服务器.网络系统.安全等领域.从而使得掌握在 Linux环境下的开发技术,成为 ...
- 嵌入式linux设备驱动程序是,嵌入式Linux设备驱动开发之:按键驱动程序实例-嵌入式系统-与非网...
11.6 按键驱动程序实例 11.6.1 按键工作原理 LED和蜂鸣器是最简单的GPIO的应用,都不需要任何外部输入或控制.按键同样使用GPIO接口,但按键本身需要外部的输入,即在驱动程序中要处理 ...
最新文章
- php error_reporting 详解
- disruptor框架为什么不流行_Java并发编程框架Disruptor
- 保护SQL Server数据库的十大绝招
- Android Activity无法启动。
- 未来计算机控制器趋势,未来DCS控制系统技术发展4大趋势
- Python之struct介绍及详解(与C/C++通信结构体的交互)
- python写tcp服务器_用Python实现一个简单的多线程TCP服务器的教程
- tomcat启动时报错ports are invalid,默认端口不要为-1
- 第九届蓝桥杯决赛大学A组
- 挖金矿 Java实现
- 中学生物教学杂志中学生物教学杂志社中学生物教学编辑部2022年第15期目录
- 汕尾python高手_放飞梦想,不问所得
- 求解立方根 java
- 风冷精密空调安装规范
- css div自适应高度height
- (对抖音程序员隔空笔芯)抖音解析去水印iteminfo接口被封死,再放一个最新的去水印原理,无视频长度限制
- Win10计算机窗口空白,Win10系统下启用或关闭windows功能打开后显示空白如何解决...
- MySql展示表字段
- 百度地图根据经纬度判断点是否在行政区内
- delphi 获取本机IP地址和MAC地址