热门标签 | HotTags
当前位置:  开发笔记 > 编程语言 > 正文

关于内核编译

字体大小:大中小博文linux设备驱动归纳总结(一):内核的相关基础概念(2012-07-1916:42)标签:转载分类&#
字体大小:大 中 小博文
linux设备驱动归纳总结(一):内核的相关基础概念 (2012-07-19 16:42)
标签:  转载  分类: linux内核与驱动

原文地址:linux设备驱动归纳总结(一):内核的相关基础概念 作者:diytvgy


linux设备驱动归纳总结(一):内核的相关基础概念

 

xxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxx


.linux设备驱动的作用

内核:用于管理软硬件资源,并提供运行环境。如分配4G虚拟空间等。

linux设备驱动:是连接硬件和内核之间的桥梁。

 

linux系统按个人理解可按下划分:

 

应用层:包括POSIX接口,LIBC,图形库等,用于给用户提供访问内核的接口。属于用户态,ARM运行在用户模式(usr)或者系统模式(sys)下。

内核层:应用程序调用相关接口后,会通过系统调用,执行SWI指令切换ARM的工作模式到超级用户(svc)模式下,根据用户函数的要求执行相应的操作。

硬件层:硬件设备,当用户需要操作硬件时,内核会根据驱动接口操作硬件设备

 

图结构如下:

 

 

举一个相对比较邪恶的类比:

在深圳的酒店经常会在门缝看到一些卡片,上面说可以通过打电话送货上门提供某中服务。


xxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxx


.内核代码树介绍

 

linux-2.6.29

|-arch : 包含和硬件体系结构相关的代码

|-block : 硬盘调度算法,不是驱动

|-firmware : 固件,如BOIS

|-Documentation: 标准官方文档

|-dirver : linux设备驱动

|-fs : 内核所支持的文件体系

|-include :头文件。linux/module.h linux/init.h 常用库。

|-init :库文件代码,C库函数在内核中的实现。

init/main.c ->start_kernel->内核执行第一条代码

|-ipc : 进程件通信

|-mm :内存管理

|-kernel : 内核核心部分,包括进程调度等

|-net :网络协议

|-sound : 所有音频相关

|

 

其中,跟设备驱动有关并且经常查阅的文件夹有:

init

include : linux, asm-arm

drivers:

arch:

 

xxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxx


.内核补丁:

补丁一般都是基于某个版本内核生成的,用于升级旧内核。

打补丁需要注意:

1.对应版本的补丁只能用于对应版本的内核。

2.如果在已打补丁的内核再打补丁,需要先卸载原来补丁。

打补丁的方法:

1.制作补丁:

diff -Nur linux-2.6.30/ linux-2.6.30.1/ > linux-2.6.30.1.patch

//N为新加的文件全部修改

//linux-2.6.30 旧版本

//linux-2.6.30.1 新版本

//目标补丁

2.打补丁:

cd linux-2.6.30 //!!注意在原文件夹的目录中打补丁

patch -p1 <../linux-2.6.30.1.patch //-p1是忽略一级目录

3.恢复&#xff1a;

cd linux-2.6.30 //&#xff01;&#xff01;注意在原文件夹的目录中打补丁

patch -R <../linux-2.6.30.1.patch //撤销补丁


xxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxx


.内核中的Makefile&#xff1a;

对于内核&#xff0c;Makefile分为5类&#xff1a;

Documentation/kbuild/makefiles.txt描述如下&#xff1a;

50 The Makefiles have five parts:

51

52 Makefile Makefile&#xff0c;控制内核的编译

53 .config 内核配置文件&#xff0c;配置内核时生成&#xff0c;如make menuconfig

54 arch/$(ARCH)/Makefile 对应体系结构的Makefile

55 scripts/Makefile.* Makefile共用的规则

56 kbuild Makefiles 各子目录下的Makefile&#xff0c;被上层的Makefile调用。

 

简单来说&#xff0c;编译内核会执行以下两步骤&#xff0c;它们分别干了以下的事情。

1一般的&#xff0c;我们会拷贝一个对应体系结构的配置文件到主目录下并改名为 .config&#xff0c;这样就在make menuconfig生成的图形配置中已经有了一些默认的配置&#xff0c;减少用户的劳动量。不过这一步不做也没关系的。

2.make menuconfig

2.1、由总Makefile决定编译的体系结构(ARCH). 编译工具(CROSS_COMPILE)&#xff0c;并知道需要进去哪些内核根下的哪些目录进行编译。

2.2、由arch/$(ARCH)/Makefile,决定arch/$(ARCH)下还有的哪些目录和文件需要编译。

2.3、知道了需要编译的目录后&#xff0c;递归的进入哪些目录下&#xff0c;读取每一个Kconfig的信息&#xff0c;生成了图形配置的界面。

2.4、通过我们在图形配置界面中选项为[*][M]或者[]

2.5、保存并退出配置&#xff0c;会根据配置生成一份新的配置文件.config&#xff0c;并在同时生成include/config/auto.conf&#xff08;这是.config的去注释版&#xff09;。文件里面保存着CONFIG_XXXX等变量应该取y还是取m

3.make

3.1、根据Makefile包含的目录和配置文件的要求&#xff0c;进去个子目录进行编译&#xff0c;最后会在各子目录下生成一个.o或者.a文件&#xff0c;然后总Makefile指定的连接脚本 arch/$(ARCH)/kernel/vmlinux.lds生成vmlinux&#xff0c;并通过压缩编程bzImage&#xff0c;或者按要求在对应的子目录下编译成模块。。

 

但是&#xff0c;具体是怎么生成配置文件的呢&#xff1f;

注&#xff1a;我使用的内核是2.6.29

1.在总Makefile中&#xff0c;根据以下语句进入需要编译的目录

470 # Objects we will link into vmlinux / subdirs we need to visit

471 init-y :&#61; init/

472 drivers-y :&#61; drivers/ sound/ firmware/

473 net-y :&#61; net/

474 libs-y :&#61; lib/

475 core-y :&#61; usr/

476 endif # KBUILD_EXTMOD

639 core-y &#43;&#61; kernel/ mm/ fs/ ipc/ security/ crypto/ block/

上面说明了&#xff0c;根目录下的initdriversoundfirmwarenetlibusr等目录&#xff0c;在编译时都会进去读取目录下的Makefile并进行编译。

 

2.在总Makefile中包含的目录还是不够的&#xff0c;内核还需要根据对应的CPU体系架构&#xff0c;

决定还需要将哪些子目录将要编译进内核。在总Makefile中有一个语句&#xff1a;

529 include $(srctree)/arch/$(SRCARCH)/Makefile //在这里&#xff0c;我定义SRCARCH &#61; arm

可以看出&#xff0c;在总Makefile中进去读取相应体系结构的Makefile->arch/$(SRCARCH)/Makefile

 

arch/$(SRCARCH)/Makefile中指定arch/$(SRCARCH)路径下的哪些子目录需要被编译。

arch/arm/Makefile 下&#xff1a;

95 head-y :&#61; arch/arm/kernel/head$(MMUEXT).o arch/arm/kernel/init_task.o

187 # If we have a machine-specific directory, then include it in the build.

188 core-y &#43;&#61; arch/arm/kernel/ arch/arm/mm/ arch/arm/common/

189 core-y &#43;&#61; $(machdirs) $(platdirs)

190 core-$(CONFIG_FPE_NWFPE) &#43;&#61; arch/arm/nwfpe/

191 core-$(CONFIG_FPE_FASTFPE) &#43;&#61; $(FASTFPE_OBJ)

192 core-$(CONFIG_VFP) &#43;&#61; arch/arm/vfp/

193

194 drivers-$(CONFIG_OPROFILE) &#43;&#61; arch/arm/oprofile/

195

196 libs-y :&#61; arch/arm/lib/ $(libs-y)

上面看到&#xff0c;指定需要进入arch/arm/kernel/arch/arm/mm/arch/arm/common/ 等目录编译&#xff0c;至于core-ycore-$(CONFIG_FPE_NWFPE)这些是什么东西呢&#xff1f;


其中&#xff0c;y表示编译成模块&#xff0c;m表示编译进内核(上面没有&#xff0c;因为默认情况下ARM全部编译进内核)&#xff0c;但$(CONFIG_OPROFILE)又是什么呢&#xff1f;这些是根据用户在make menuconfig中设置后&#xff0c;生成的值赋给了CONFIG_OPROFILE


3.make menuconfig后的配置信息是怎么来的&#xff1f;

这是由各子目录下的Kconfig提供选项功用户选择并配置。

arch/arm/Kconfig。所有的配置都是根据arch/$(ARCH)/Kconfig文件通过Kconfig的语法source读取各个包含的子目录Kconfig来生成一个配置界面。每个Makefile目录下都有一个对应的Kconfig文件&#xff0c;用于生成配置界面来给用户决定内核如何配置&#xff0c;配置后会确定一个。 CONFIG_XXX的的值&#xff08;如上面的CONFIG_OPROFILE&#xff09;&#xff0c;来决定编译进内核&#xff0c;还是编译成模块或者不编译。

如在arch/arm/Kconfig下&#xff1a;

595 source "arch/arm/mach-clps711x/Kconfig"

596

597 source "arch/arm/mach-ep93xx/Kconfig"

598

599 source "arch/arm/mach-footbridge/Kconfig"

600

601 source "arch/arm/mach-integrator/Kconfig"

602

603 source "arch/arm/mach-iop32x/Kconfig"

604

605 source "arch/arm/mach-iop33x/Kconfig"

这些就是用来指定&#xff0c;需要读取以下目录下的Kconfig文件来生成一个使用make menuconfig时的配置界面。

至于子目录下的Kconfig是怎么样的&#xff0c;待会介绍。

总结Kconfig的作用&#xff1a;

3.1.make menuconfig下可以配置选项;

3.2..config中确定CONFIG_XXX的的值。


4.只是读取以上的两个Makefile还是不够了&#xff0c;内核还会把包含的子目录一层一层的读取它里面的MakefileKconfig


上面啰啰嗦嗦地讲了这么久&#xff0c;无非就是想说&#xff0c;内核的编译并不是一个Makefile搞定的&#xff0c;需要通过根目录下的总Makefile来包含一下子Makefile&#xff08;不管是根目录下的子目录还是/arch/arm中的子目录&#xff09;。而Kconfig&#xff0c;为用户提供一个交互界面来选择如何配置并生成配置选项。


xxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxx


五、子目录下的MakefileKconfig


上面我一直介绍的都是两个比较大的Makefile——Makefilearch/$(ARCH)/Makefile。接下来看一下实例。


一、makefile中&#xff0c;y表示编译进内核&#xff0c;m表示编译成模块&#xff0c;不写代表不编译。所以&#xff0c;配置最简单的方法就是&#xff0c;直接修改子目录的Makefile

先看看arch/arm/Makefile&#xff1a;

/*arch/arm/mach-s3c2440/Makefile */

12 obj-$(CONFIG_CPU_S3C2440) &#43;&#61; s3c2440.o dsc.o

13 obj-$(CONFIG_CPU_S3C2440) &#43;&#61; irq.o

14 obj-$(CONFIG_CPU_S3C2440) &#43;&#61; clock.o //配置2440的时钟进入模块

15 obj-$(CONFIG_S3C2440_DMA) &#43;&#61; dma.o

如果我要取消s3c2440的时钟(当然这是必须要开的&#xff0c;只是举例)。可以直接修改arch/arm/mach-s3c2440/Makefile obj-$( CONFIG_CPU_S3C2440) &#43;&#61; clock.o改为

obj- &#43;&#61; clock.o

如果你想编译成模块也可以修改成&#xff1a;

obj-m &#43;&#61; clock.o

在这里 CONFIG_CPU_S3C2440的值默认是y&#xff0c;所以内核是要将时钟编译进内核的。也许有人会问&#xff0c;那我直接修改 CONFIG_CPU_S3C2440的值为m不就可以将时钟编译成模块了&#xff0c;何必修改Makefile这么麻烦呢&#xff1f;的确是这样&#xff0c;只要我们通过在”make menuconfig”的界面中配置后就能够改变 CONFIG_CPU_S3C2440的值。接下来看看如何实现。


二、在一般的编译内核时&#xff0c;我们都是通过”make menuconfig”进入图形界面面配置的&#xff0c; 接下来我实现一下如何将一个选项加入到图形配置界面中。

看看具体实现的步骤&#xff1a;

以下的执行环境是在PC机上&#xff0c;我使用的内核是linux-2.6.29&#xff1a;

2.1.进入内核目录

cd linux-2.6.29

2.2. driver目录下模拟一个名为test1驱动的文件夹

mkdir driver/test1

2.3. 在目录下随便些一个C文件&#xff0c;只要不报错。

vim test1.c

我的test1.c如下&#xff1a;

1 void foo()

2 {

3   ;

4 }

2.4vim Makefile //在目录下编写一个简单的Makefile

Makefile文件编写如下&#xff1a;

obj-$(CONFIG_TEST1) &#43;&#61; test1.o

CONFIG_TEST1是决定test1是否编译进内核或者编译成模块的。这就是通过同一目录下的Kconfig来在配置界面中生成选项&#xff0c;由用户在make menuconfig中选择。

2.5所以还要同一目录下写一个Kconfig&#xff1a;

vim Kconfig

Kconfig修改如下&#xff1a;

menu "test1 driver here" //这是在图形配置显示的

config TEST1

bool "xiaobai test1 driver" //这同样也是在图形配置显示的

help

This is test1 //这个也是在图形配置显示的。

说白了&#xff0c;就是在图形配置的driver下多了一个配置选项&#xff0c;用户配置后将 CONFIG_TEST1的值存放在.config中&#xff0c;Makefile通过读取.config的去注释版include/config/auto.conf读取到CONFIG_TEST的值&#xff0c;再进行编译。


但是&#xff0c;以上几步还不能达到目的&#xff0c;因为虽然在总Makefile中已经包含了目录driver,但是driver目录的Makefile中并没有包含test目录。因此需要在driver/Makefile中添加&#xff1a;

103 obj-$(CONFIG_PPC_PS3) &#43;&#61; ps3/

104 obj-$(CONFIG_OF) &#43;&#61; of/

105 obj-$(CONFIG_SSB) &#43;&#61; ssb/

106 obj-$(CONFIG_VIRTIO) &#43;&#61; virtio/

107 obj-$(CONFIG_STAGING) &#43;&#61; staging/

108 obj-y &#43;&#61; platform/

109 obj-$(CONFIG_TEST1) &#43;&#61; test1/ //这是我添加的

虽然Makefile中已经包含了&#xff0c;但这样还是不行。因为当需要配置ARM时&#xff0c; ARM结构下的Kconfig并没有包含testKconfig这样的话就不会出现在图形配置界面中&#xff0c;因此在arch/arm/Kconfig中添加&#xff1a;

1230 menu "Device Drivers" //要在Device Drivers这个选项里面添加

1231

1232 source "drivers/base/Kconfig"

1233

1234 source "drivers/connector/Kconfig"

。。。。。。。。

1330 source "drivers/test/Kconfig" //这是我添加的

1331

1332 endmenu

 

大功告成&#xff01;

这样&#xff0c;make menuconfig界面写的Driver Devices下就多了一个 "test1 friver here"的目录&#xff0c;里面有一个配置选项"xiaobai test1 driver"

 

 

Kconfig文件的语法在documentation/kbuild/kconfig-language.txt文件中有详细的讲解&#xff0c;上面我只是简单实现了一下&#xff0c;都是皮毛。


xxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxx


.内核和模块的编译

 

编译内核很简单&#xff0c;只需要配置完毕后执行make命令&#xff0c;将指定的文件编译进内核

bzImage或者编译成模块。

make &#61; make bzImage &#43; make modules

因此如果值编译内核&#xff0c;即只编译配置文件中-y选项&#xff0c;可以直接用命令

make bzImage

如果值编译模块&#xff0c;即只编译配置文件中的-m选项&#xff0c;可以之直接使用命令

make modules

模块可以编译当然也可以清除&#xff0c;使用命令

make modules clean

如果只想单独编译一个模块&#xff0c;可以使用命令

make M&#61;drivers/test/ modules //只单独编译drivers/test中的.ko

make M&#61;drivers/test/ modules clean //清除

上面的是在内核目录下的操作&#xff0c;但当我写驱动时&#xff0c;我并不可能在内核目录下编

写&#xff0c;但我编译时却要依赖内核中的规则和Makefile&#xff0c;所以就有了以下的方法&#xff0c;

同时这也是一般的编写驱动时Makefile的格式。

指定内核Makefile并单独编译

make -C /root/linux-2.6.29 M&#61;&#96;pwd&#96; module

make -C /root/linux-2.6.29 M&#61;&#96;pwd&#96; module clean

//-C 指定内核Makefile的路径&#xff0c;可以使用相对路径。

//-M 指定要编译的文件的路径&#xff0c;同样课使用相对路径。

编译生成的模块可以指定存放的目录

make -C /root/linux-2.6.29 M&#61;&#96;pwd&#96; modules_install INSTALL_MOD_PATH&#61;/nfsroot


xxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxx


七、总结


说了这么久估计都说糊涂了&#xff0c;其实我只是想表达一下内核编译时大体上究竟是怎么样的一个过程。

我以编译S3C2440的内核为例再说一遍&#xff1a;

1一般我们会想将一份S3C2440的默认配置拷贝到内核跟目录下并改名为.config

2.make menuconfig

2.1、由总Makefile决定编译的体系结构(ARCH). 编译工具(CROSS_COMPILE)&#xff0c;并知道需要进去哪些内核根下的哪些目录进行编译。

2.2、由arch/$(ARCH)/Makefile,决定arch/$(ARCH)下还有的哪些目录和文件需要编译。

2.3、知道了需要编译的目录后&#xff0c;递归的进入哪些目录下&#xff0c;读取每一个Kconfig的信息&#xff0c;生成了图形配置的界面。

2.4、通过我们在图形配置界面中选项为[*][M]或者[]

2.5、保存并退出配置&#xff0c;会根据配置生成一份新的配置文件.config&#xff0c;并在同时生成include/config/auto.conf&#xff08;这是.config的去注释版&#xff09;。文件里面保存着CONFIG_XXXX等变量应该取y还是取m

3.make

3.1、根据Makefile包含的目录和配置文件的要求&#xff0c;进去个子目录进行编译&#xff0c;最后会在各子目录下生成一个.o或者.a文件&#xff0c;然后总Makefile指定的连接脚本 arch/$(ARCH)/kernel/vmlinux.lds生成vmlinux&#xff0c;并通过压缩编程bzImage&#xff0c;或者按要求在对应的子目录下编译成模块

xxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxx

源代码&#xff1a; 1st_kernel.rar   


转:https://www.cnblogs.com/superlcc/archive/2012/08/07/2626582.html



推荐阅读
  • 32位ubuntu编译android studio,32位Ubuntu编译Android 4.0.4问题
    问题一:在32位Ubuntu12.04上编译Android4.0.4源码时,出现了关于emulator的错误,关键是其Makefile里的 ... [详细]
  • 本文概述了JNI的原理以及常用方法。JNI提供了一种Java字节码调用C/C++的解决方案,但引用类型不能直接在Native层使用,需要进行类型转化。多维数组(包括二维数组)都是引用类型,需要使用jobjectArray类型来存取其值。此外,由于Java支持函数重载,根据函数名无法找到对应的JNI函数,因此介绍了JNI函数签名信息的解决方案。 ... [详细]
  • SpringBoot整合SpringSecurity+JWT实现单点登录
    SpringBoot整合SpringSecurity+JWT实现单点登录,Go语言社区,Golang程序员人脉社 ... [详细]
  • C#设计模式之八装饰模式(Decorator Pattern)【结构型】
    一、引言今天我们要讲【结构型】设计模式的第三个模式,该模式是【装饰模式】,英文名称:DecoratorPattern。我第一次看到这个名称想到的是另外一个词语“装修”,我就说说我对“装修”的理 ... [详细]
  • 用户视图(查看运行状态或其他参数)系统视图(配置设备的系统参数)system-viewEntersystemview,returnuservi ... [详细]
  • TerraformVersionTerraformv0.9.11AffectedResource(s)Pleas ... [详细]
  • 每当我尝试使用NEON16位浮点内在函数时都会收到此错误。我没有遇到其他数据类型内在函数的任何问题。是否可以在Android上使用NEON16位浮点内在函数? ... [详细]
  • 本文主要解析了Open judge C16H问题中涉及到的Magical Balls的快速幂和逆元算法,并给出了问题的解析和解决方法。详细介绍了问题的背景和规则,并给出了相应的算法解析和实现步骤。通过本文的解析,读者可以更好地理解和解决Open judge C16H问题中的Magical Balls部分。 ... [详细]
  • 本文介绍了在CentOS上安装Python2.7.2的详细步骤,包括下载、解压、编译和安装等操作。同时提供了一些注意事项,以及测试安装是否成功的方法。 ... [详细]
  • 海马s5近光灯能否直接更换为H7?
    本文主要介绍了海马s5车型的近光灯是否可以直接更换为H7灯泡,并提供了完整的教程下载地址。此外,还详细讲解了DSP功能函数中的数据拷贝、数据填充和浮点数转换为定点数的相关内容。 ... [详细]
  • 本文介绍了利用ARMA模型对平稳非白噪声序列进行建模的步骤及代码实现。首先对观察值序列进行样本自相关系数和样本偏自相关系数的计算,然后根据这些系数的性质选择适当的ARMA模型进行拟合,并估计模型中的位置参数。接着进行模型的有效性检验,如果不通过则重新选择模型再拟合,如果通过则进行模型优化。最后利用拟合模型预测序列的未来走势。文章还介绍了绘制时序图、平稳性检验、白噪声检验、确定ARMA阶数和预测未来走势的代码实现。 ... [详细]
  • 如何使用PLEX播放组播、抓取信号源以及设置路由器
    本文介绍了如何使用PLEX播放组播、抓取信号源以及设置路由器。通过使用xTeve软件和M3U源,用户可以在PLEX上实现直播功能,并且可以自动匹配EPG信息和定时录制节目。同时,本文还提供了从华为itv盒子提取组播地址的方法以及如何在ASUS固件路由器上设置IPTV。在使用PLEX之前,建议先使用VLC测试是否可以正常播放UDPXY转发的iptv流。最后,本文还介绍了docker版xTeve的设置方法。 ... [详细]
  • Mono为何能跨平台
    概念JIT编译(JITcompilation),运行时需要代码时,将Microsoft中间语言(MSIL)转换为机器码的编译。CLR(CommonLa ... [详细]
  • 原文地址http://balau82.wordpress.com/2010/02/28/hello-world-for-bare-metal-arm-using-qemu/最开始时 ... [详细]
  • 直接从网上下载redis当然你也可以直接从别的地方拿过来直接放在redis中[root@iZ2zedckzf8nczp6xshv4mZ]#wgethttp:download ... [详细]
author-avatar
Stupid锋_891
这个家伙很懒,什么也没留下!
PHP1.CN | 中国最专业的PHP中文社区 | DevBox开发工具箱 | json解析格式化 |PHP资讯 | PHP教程 | 数据库技术 | 服务器技术 | 前端开发技术 | PHP框架 | 开发工具 | 在线工具
Copyright © 1998 - 2020 PHP1.CN. All Rights Reserved | 京公网安备 11010802041100号 | 京ICP备19059560号-4 | PHP1.CN 第一PHP社区 版权所有