C
C运行库(glibc, GNU C Library)
参考C运行库实现https://ftp.gnu.org/gnu/glibc/glibc-2.0.6.tar.gz。
Windows下VC C运行库
msvcrt.dll
msvcrt20.dll
msvcrt40.dll
MSVCRTD.DLL
以上不同dll应用于不同的vc版本,MSVCRTD.DLL用于debug。
msvcr100.dll
msvcr100_clr0400.dll
msvcr100d.dll
msvcr110.dll
msvcr71.dll
以上不同dll应用于不同的vc版本,msvcr100d.dll用于debug。
静态链接库
LIBC.LIB
LIBCD.LIB
LIBCMT.LIB
LIBCMTD.LIB
C++运行库
LIBCP.LIB
LIBCPD.LIB
LIBCPMT.LIB
LIBCPMTD.LIB
LIBCPD.LIB用于debug。
LIBCPMT.LIB,LIBCPMTD.LIB为多线程实现,LIBCPMTD.LIB用于debug。
C/C++ entry
写道
mainCRTStartup(void)
wmainCRTStartup(void)
WinMainCRTStartup(void)
wWinMainCRTStartup(void)
Purpose:
These routines do the C runtime initialization, call the appropriate
user entry function, and handle termination cleanup. For a managed
app, they then return the exit code back to the calling routine, which
is the managed startup code. For an unmanaged app, they call exit and
never return.
Entry:
Function: User entry called:
mainCRTStartup main
wmainCRTStartup wmain
WinMainCRTStartup WinMain
wWinMainCRTStartup wWinMain
malloc
#define mALLOc malloc
#if __STD_C
Void_t* mALLOc(size_t bytes)
#else
Void_t* mALLOc(bytes) size_t bytes;
#endif
{
arena *ar_ptr;
INTERNAL_SIZE_T nb; /* padded request size */
mchunkptr victim;
#if defined(_LIBC) || defined(MALLOC_HOOKS)
if (__malloc_hook != NULL) {
Void_t* result;
result = (*__malloc_hook)(bytes);
return result;
}
#endif
nb = request2size(bytes);
arena_get(ar_ptr, nb);
if(!ar_ptr)
return 0;
victim = chunk_alloc(ar_ptr, nb);
(void)mutex_unlock(&ar_ptr->mutex);
if(!victim) {
/* Maybe the failure is due to running out of mmapped areas. */
if(ar_ptr != &main_arena) {
(void)mutex_lock(&main_arena.mutex);
victim = chunk_alloc(&main_arena, nb);
(void)mutex_unlock(&main_arena.mutex);
}
if(!victim) return 0;
}
return chunk2mem(victim);
}
#define arena_get(ptr, size) do { \
Void_t *vptr = NULL; \
ptr = (arena *)tsd_getspecific(arena_key, vptr); \
if(ptr && !mutex_trylock(&ptr->mutex)) { \
THREAD_STAT(++(ptr->stat_lock_direct)); \
} else \
ptr = arena_get2(ptr, (size)); \
} while(0)
static arena *
#if __STD_C
arena_get2(arena *a_tsd, size_t size)
#else
arena_get2(a_tsd, size) arena *a_tsd; size_t size;
#endif
{
arena *a;
heap_info *h;
char *ptr;
int i;
unsigned long misalign;
if(!a_tsd)
a = a_tsd = &main_arena;
else {
a = a_tsd->next;
if(!a) {
/* This can only happen while initializing the new arena. */
(void)mutex_lock(&main_arena.mutex);
THREAD_STAT(++(main_arena.stat_lock_wait));
return &main_arena;
}
}
/* Check the global, circularly linked list for available arenas. */
do {
if(!mutex_trylock(&a->mutex)) {
THREAD_STAT(++(a->stat_lock_loop));
tsd_setspecific(arena_key, (Void_t *)a);
return a;
}
a = a->next;
} while(a != a_tsd);
/* Nothing immediately available, so generate a new arena. */
h = new_heap(size + (sizeof(*h) + sizeof(*a) + MALLOC_ALIGNMENT));
if(!h)
return 0;
a = h->ar_ptr = (arena *)(h+1);
for(i=0; i<NAV; i++)
init_bin(a, i);
a->next = NULL;
a->size = h->size;
tsd_setspecific(arena_key, (Void_t *)a);
mutex_init(&a->mutex);
i = mutex_lock(&a->mutex); /* remember result */
/* Set up the top chunk, with proper alignment. */
ptr = (char *)(a + 1);
misalign = (unsigned long)chunk2mem(ptr) & MALLOC_ALIGN_MASK;
if (misalign > 0)
ptr += MALLOC_ALIGNMENT - misalign;
top(a) = (mchunkptr)ptr;
set_head(top(a), (((char*)h + h->size) - ptr) | PREV_INUSE);
/* Add the new arena to the list. */
(void)mutex_lock(&list_lock);
a->next = main_arena.next;
main_arena.next = a;
(void)mutex_unlock(&list_lock);
if(i) /* locking failed; keep arena for further attempts later */
return 0;
THREAD_STAT(++(a->stat_lock_loop));
return a;
}
内嵌汇编
VC下内嵌汇编的例子:(Intel汇编语法)
// cmpxchg op1, op2
//
// cmpxchg use al, ax, eax, rax as output.
// if al, ax, eax, rax equals op1, op2 loaded into op1, else, load op2 into al, ax, eax, rax.
short cmpxchg_s(short v_exchange, short* v_ptr, short v_compare)
{
__asm mov edx, v_ptr; // mov v_ptr to edx
__asm mov ax, v_compare; // mov v_compare to ax
__asm mov cx, v_exchange; // mov v_exchange to cx
__asm cmpxchg word ptr [edx], cx;
// below comment code use to get the output of cmpxchg and return. but
// it's is needless. by defult, the last statement's result returned.
// in this case, the output of cmpxchg is returned.
//__asm mov v_exchange, ax;
//return v_exchange;
}
int cmpxchg_sx(short v_exchange, short* v_ptr, short v_compare)
{
return cmpxchg_s(v_exchange, v_ptr, v_compare) == v_compare;
}
int cmpxchg(int v_exchange, int* v_ptr, int v_compare)
{
__asm mov edx, v_ptr; // mov v_ptr to edx
__asm mov eax, v_compare; // mov v_compare to ax
__asm mov ecx, v_exchange; // mov v_exchange to cx
__asm cmpxchg dword ptr [edx], ecx;
// below comment code use to get the output of cmpxchg and return. but
// it's is needless. by defult, the last statement's result returned.
// in this case, the output of cmpxchg is returned.
//__asm mov v_exchange, eax;
//return v_exchange;
}
int cmpxchg_x(int v_exchange, int* v_ptr, int v_compare)
{
return cmpxchg(v_exchange, v_ptr, v_compare) == v_compare;
}
gnulib
参考另一篇文章:https://lobin.iteye.com/blog/609813
ANSI C grammar, Lex specification,http://www.lysator.liu.se/c/ANSI-C-grammar-l.html
https://port70.net/~nsz/c/c11
Programming languages -- C,https://port70.net/~nsz/c/c11/n1570.html
ANSI C Specification,http://eli-project.sourceforge.net/c_html/c.html
Rationale for American National Standard for Information Systems - Programming Language - C,http://www.lysator.liu.se/c/rat/title.html
select & poll
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <sys/socket.h>
#include <poll.h>
#include <sys/time.h>
#include <netinet/in.h>
#include <errno.h>
#include <unistd.h>
#define PORT 6002
//最多处理的connect
#define BACKLOG 5
//当前的连接数
int currentClient = 0;
//数据接受 buf
#define REVLEN 10
char recvBuf[REVLEN];
#define OPEN_MAX 1024
struct pollfd* _fd_init(struct pollfd* const fd_set)
{
struct pollfd *poll_fd_set;
if (fd_set == NULL)
{
poll_fd_set = (struct pollfd *) malloc(OPEN_MAX * sizeof(struct pollfd));
}
else
{
poll_fd_set = fd_set;
}
return poll_fd_set;
}
void _fd_zero(struct pollfd* fd_set)
{
for(int i = 0; i < OPEN_MAX; i++)
{
fd_set[i].fd = -1;
}
}
void _fd_set(int fd, struct pollfd* const fd_set)
{
fd_set[0].fd = fd;
fd_set[0].events = POLLIN; //POLLRDNORM;
}
class SocketException
{
private:
int errorCode;
char *message;
public:
SocketException(int errorCode)
{
this->errorCode = errorCode;
this->message = NULL;
}
SocketException(int errorCode, char *message)
{
this->errorCode = errorCode;
this->message = message;
}
int getCode()
{
return errorCode;
}
char* getMessage()
{
return message;
}
};
class HardSocket
{
private:
int sfd;
public:
HardSocket() throw (SocketException)
{
/**
* On success, a file descriptor for the new socket is returned. On
* error, -1 is returned, and errno is set appropriately.
*/
int fd = socket(AF_INET, SOCK_STREAM, 0);
//fd = -1;
if (fd == -1)
{
int error = errno;
throw SocketException(error);
}
this->sfd = fd;
}
~HardSocket()
{
printf("~HardSocket\n");
}
int getPlainSocket()
{
return this->sfd;
}
void bind(int port) throw (SocketException)
{
struct sockaddr_in server_addr;
bzero(&server_addr, sizeof(server_addr));
server_addr.sin_family = AF_INET;
server_addr.sin_port = htons(port);
server_addr.sin_addr.s_addr = htonl(INADDR_ANY);
int error = ::bind(this->sfd, (struct sockaddr*) &server_addr, sizeof(server_addr));
if(error == -1)
{
error = errno;
throw SocketException(error);
}
}
void listen() throw (SocketException)
{
this->listen(50);
}
void listen(int backlog) throw (SocketException)
{
int error = ::listen(this->sfd, backlog);
if(error == -1)
{
error = errno;
throw SocketException(error);
}
}
};
class Poll
{
};
int main()
{
HardSocket *socket = NULL;
try
{
socket = new HardSocket();
socket->bind(PORT);
socket->listen();
}
catch (SocketException e)
{
printf("socket error %d\n", e.getCode());
// 有问题, 这里实例化一个socket对象时抛出异常的话, 析构函数不会被调用以释放对象资源
//sleep(5000);
return 1;
}
/**
* struct pollfd {
* int fd; // file descriptor
* short events; // requested events
* short revents; // returned events
* };
*/
struct pollfd clientfd[OPEN_MAX];
_fd_zero(clientfd);
_fd_set(socket->getPlainSocket(), clientfd);
int nfds = 0;
while(1)
{
int timeout = 3000;
int error = poll(clientfd, nfds+1, timeout);
if(error < 0) // On error, -1 is returned, and errno is set appropriately.
{
printf("select error %d\n", errno);
break;
}
else if(error == 0) // A value of 0 indicates that the call timed out and no file descriptors were ready.
{
printf("timeout ...\n");
continue;
}
for(int i = 0; i <= nfds; i++)
{
if(clientfd[i].fd < 0)
{
continue;
}
if (clientfd[i].fd == socket->getPlainSocket())
{
/**
* Case indicates that one event that data to read (POLLIN or POLLRDNORM) contained.
*
* // The following values are defined by XPG4.
* #define POLLRDNORM POLLIN
*/
if (clientfd[i].revents & POLLIN)
{
printf("clientfd[%d].revents & POLLIN %d(0x%x) clientfd[%d].revents %d(0x%x), POLLIN %d(0x%x)\n",
i,
clientfd[i].revents & POLLIN,
clientfd[i].revents & POLLIN,
i,
clientfd[i].revents,
clientfd[i].revents,
POLLIN,
POLLIN);
int sockSvr = accept(socket->getPlainSocket(), NULL, NULL);//(struct sockaddr*)&client_addr
if(sockSvr == -1)
{
printf("accpet error\n");
}
else
{
currentClient++;
}
for(i=0; i<OPEN_MAX; i++)
{
if(clientfd[i].fd<0)
{
clientfd[i].fd = sockSvr;
break;
}
}
if(i == OPEN_MAX)
{
printf("too many connects\n");
return -1;
}
clientfd[i].events = POLLIN;//POLLRDNORM;
if(i > nfds)
{
nfds = i;
}
}
}
else if (clientfd[i].revents & (POLLIN | POLLERR)) // POLLRDNORM
{
printf("clientfd[%d].revents & (POLLIN | POLLERR) %d(0x%x) clientfd[%d].revents %d(0x%x), POLLIN %d(0x%x), POLLERR %d(0x%x)\n",
i,
clientfd[i].revents & (POLLIN | POLLERR),
clientfd[i].revents & (POLLIN | POLLERR),
i,
clientfd[i].revents,
clientfd[i].revents,
POLLIN,
POLLIN,
POLLERR,
POLLERR);
int recvLen = 0;
if(recvLen != REVLEN)
{
while(1)
{
printf("recv....\n");
//recv数据
int bytes = recv(clientfd[i].fd, (char *) recvBuf+recvLen, REVLEN - recvLen, 0);
if(bytes == 0)
{
printf("stream socket %d peer has performed an orderly shutdown\n", clientfd[i].fd);
clientfd[i].fd = -1;
recvLen = 0;
break;
}
else if(bytes == -1)
{
printf("receive a message from a socket error %d\n", errno);
clientfd[i].fd = -1;
recvLen = 0;
break;
}
//数据接受正常
recvLen = recvLen + bytes;
if(recvLen < REVLEN)
{
continue;
}
else
{
//数据接受完毕
printf("buf = %s\n", recvBuf);
//close(client[i].fd);
//client[i].fd = -1;
recvLen = 0;
break;
}
}
}
}
}
}
return 0;
}
select
#pragma comment(lib,"ws2_32.lib")
#include <stdio.h>
#include <string.h>
#include <WINSOCK2.H>
#include <log.h>
#include "socket/HardSocket.h"
#include "socket/LightSocket.h"
#include "SelectEvent.hpp"
#define INT_SERVER_PORT 6002
#define STR_SERVER_IP "127.0.0.1"
#define INT_DATABUFFER_SIZE 100
class SelectEventBusinessHandler : public SelectEvent
{
public:
void onAccept(LightSocket &socket)
{
sockaddr_in addr = socket.getSocketAddress();
debug("handle on accept event from %s:%d(socket %d)", inet_ntoa(addr.sin_addr), ntohs(addr.sin_port), socket.getPlainSocket());
}
void onMessage(LightSocket &socket, char *bytes)
{
sockaddr_in addr = socket.getSocketAddress();
//打印接收的数据
info("recv from %s:%d\ndata:%s", inet_ntoa(addr.sin_addr), ntohs(addr.sin_port), bytes);
}
};
class Select : public HardSocket
{
private:
fd_set fd;
SelectEvent* eventHandler;
public:
Select()
{
Socket::init(AF_INET, SOCK_STREAM, 0);
eventHandler = new SelectEventBusinessHandler();
FD_ZERO(&fd);
FD_SET(this->getPlainSocket(), &fd);
}
void select()
{
fd_set fdOld = fd;
//timeval tm;
//tm.tv_sec = 0;
//tm.tv_usec = 1000;
int selected = ::select(0, &fdOld, NULL, NULL, NULL);
if (selected == SOCKET_ERROR)
{
//WSACleanup();
//printf("Faild to select sockt in server!/r/n");
int error = WSAGetLastError();
warn("select error %d", error);
Sleep(100);
}
else if (selected == 0)
{
warn("select error, time limit expired");
}
else
{
for(int i = 0;i < fd.fd_count; i++)
{
SOCKET sfd = fd.fd_array[i];
if (FD_ISSET(sfd, &fdOld))
{
//如果socket是服务器,则接收连接
if (sfd == this->getPlainSocket())
{
onAccept();
}
else //非服务器,接收数据(因为fd是读数据集)
{
onRead(sfd);
}
}
else
{
warn("socket fd %d not in fd set", sfd);
}
}
}
}
void onAccept()
{
sockaddr_in addrAccept;
int so_sockaddr_in = sizeof(sockaddr_in);
memset(&addrAccept, 0, so_sockaddr_in);
SOCKET sockAccept = ::accept(this->getPlainSocket(), (sockaddr *) &addrAccept, &so_sockaddr_in);
if (sockAccept == INVALID_SOCKET)
{
warn("accepted an invalid socket connection");
return;
}
FD_SET(sockAccept, &fd);
//FD_SET(sockAccept,&fdOld);
info("%s:%d has connected server!", inet_ntoa(addrAccept.sin_addr), ntohs(addrAccept.sin_port));
LightSocket *cs = new LightSocket(sockAccept);
cs->setSocketAddress(addrAccept);
eventHandler->onAccept(*cs);
}
void onRead(SOCKET sfd)
{
char szDataBuff[INT_DATABUFFER_SIZE];///////
memset(szDataBuff, 0, INT_DATABUFFER_SIZE);
szDataBuff[INT_DATABUFFER_SIZE - 1] = '\0';
int bytes = recv(sfd, szDataBuff, INT_DATABUFFER_SIZE - 1, 0);
sockaddr_in addr;
int so_sockaddr_in = sizeof(sockaddr_in);
memset(&addr, 0, so_sockaddr_in);
getpeername(sfd, (sockaddr *) &addr, &so_sockaddr_in);
if (bytes == SOCKET_ERROR)
{
int error = WSAGetLastError();
warn("Fail to receive data from %s:%d error %d", inet_ntoa(addr.sin_addr), ntohs(addr.sin_port), error);
closesocket(sfd);
FD_CLR(sfd, &fd);
//i--;
return;
}
if (bytes == 0)
{
//客户socket关闭
warn("%s:%d has closed!", inet_ntoa(addr.sin_addr), ntohs(addr.sin_port));
closesocket(sfd);
FD_CLR(sfd, &fd);
//i--;
}
if (bytes > 0)
{
LightSocket *cs = new LightSocket(sfd);
cs->setSocketAddress(addr);
eventHandler->onMessage(*cs, szDataBuff);
}
}
};
void main(void)
{
///*
Select *socket = new Select();
bool bReuseAddr = true;
socket->setSocketOption(SOL_SOCKET, SO_REUSEADDR, (char *) &bReuseAddr, sizeof(bReuseAddr));
//unsigned long cmd = 1;
//iResult= ioctlsocket(sockServer,FIONBIO,&cmd);
socket->bind(INT_SERVER_PORT);
socket->listen(5);
info("Start server...");
while(1)
{
socket->select();
}
WSACleanup();
}
开发自己的C运行库及标准库
https://www.iteye.com/blog/lobin-620212
相关推荐
国民经济行业分类与国际标准行业分类(ISIC+Rev.4)的对照和匹配(供参考).docx
网络助手工具(亲测好用)
1.版本:matlab2014/2019a/2024a 2.附赠案例数据可直接运行matlab程序。 3.代码特点:参数化编程、参数可方便更改、代码编程思路清晰、注释明细。 4.适用对象:计算机,电子信息工程、数学等专业的大学生课程设计、期末大作业和毕业设计。
这是一款基于jQuery+HTML5实现的微信大转盘抽奖效果源码,是一款可配置奖品抽奖的jQuery大转盘抽奖代码,可实现点击按钮转轮旋转实现抽奖功能,效果逼真自然,是一款非常优秀的特效源码
自动驾驶控制-基于MPC的速度控制仿真 matlab和simulink联合仿真,基于mpc算法的速度控制,跟踪阶跃形式的速度和正弦形式的速度。
求解 2D 和 3D 分数矢量亥姆霍兹方程,用于非刚性图像配准Matlab代码.rar
springboot103抗疫物资管理系统,含有完整的源码和报告文档
这是2023年华南理工大学机器人校内赛五山五狼—车车难造队的机械建模开源,我是本队机械建模队员杨欣然_-
将多个图像显示为子图的蒙太奇Matlab代码.rar
labview源码参考示例,可供参考学习使用
内容概要:本文档主要聚焦于2024年秋季学期高维数据分析课程的第六次作业,涉及多个关于复杂网络分析的重要概念和技术。具体涵盖:基于图论的基本证明如节点度与共同邻居的数学表达形式;对网络社区结构进行划分,并通过模态性公式评价这种划分的效果;针对实际社交网络数据集利用随机块模型和社会经济回报模型(SCORE)来进行高效的社区识别任务;以及深入探讨了矩阵的核范数及其与谱范数间的关系,强调了它们在解决现实生活中大型稀疏矩阵问题方面的重要性。 适合人群:面向正在学习数据挖掘、机器学习等相关专业的大三及以上的学生,特别是有兴趣深入了解图论及其在网络科学中具体应用的研究人员。 使用场景及目标:该文档用于帮助学生更好地掌握高维数据分析技巧,在实际项目实践中提高解决问题的能力。通过本练习可以加深对于社交网络构建、模块化计算方法的认识,同时也能加强对线性代数里关键概念的理解。这对于准备从事数据分析相关工作或进一步深造的同学非常有益。 其他说明:这份PDF材料包含了大量详细的数学推导步骤,以及具体的评分标准和作业完成时间限制(截止日期:12月29日星期日晚上十点)。为了更好地完成这门课的任务,建议配合教科书和其他参考资料一起使用,鼓励学生积极思考每个问题背后的原理。
教育培训机构劳动合同
①数据来源:自己从cnrds数据库下载的原始数据,未做任何处理 ②数据年份:不同表格不一致,很多表格比如上市公司绿色专利申请与获得都是从1991年开始的,目前除了引用里的被引用信息更新到23年11月底,其余均更新到23年底 ③数据内容:两个表格 数据清单: 各省市绿色专利申请与获得(内部是地级市) 各省市绿色专利申请情况 各省市绿色专利获得情况
毕业设计-基于Java+SpringBoot的校园互助系统微信小程序源码+数据库.zip文件 该项目是个人项目源码,项目中的源码都是经过本地编译过可运行的,都经过严格调试,确保可以运行!!!评审分达到95分以上。资源项目的难度比较适中,内容都是经过助教老师审定过的能够满足学习、使用需求,如果有需要的话可以放心下载使用。 该项目是个人项目源码,项目中的源码都是经过本地编译过可运行的,都经过严格调试,确保可以运行!!!评审分达到95分以上。资源项目的难度比较适中,内容都是经过助教老师审定过的能够满足学习、使用需求,如果有需要的话可以放心下载使用。 基于微信小程序的校园互助系统源码+数据库.zip文件 该项目是个人项目源码,项目中的源码都是经过本地编译过可运行的,都经过严格调试,确保可以运行!!!评审分达到95分以上。资源项目的难度比较适中,内容都是经过助教老师审定过的能够满足学习、使用需求,如果有需要的话可以放心下载使用。 基于微信小程序的校园互助系统源码+数据库.zip文件 该项目是个人项目源码,项目中的源码都是经过本地编译过可运行的,都经过严格调试,确保可以运
springboot128中小企业人事管理系统代码,含有完整的源码和报告文档
管理员功能 用户管理 管理各类用户(社团负责人、会员)的注册、审核、修改和删除。 设置不同角色的权限,确保系统安全性。 社团管理 创建、编辑和删除社团信息,包括社团名称、简介、活动类型等。 审核新申请的社团,并管理社团的状态(正常、暂停、注销)。 活动管理 发布、编辑和删除社团活动信息,包含活动主题、时间、地点和参与人数。 查看活动报名情况和参与者列表。 财务管理 管理社团资金,包括收支记录、预算编制和财务报表生成。 支持费用报销流程,审核并记录社团开支。 数据统计与报告 生成各类统计报告,如社团活动参与情况、会员增长趋势等。 分析社团运营状况,为决策提供数据支持。 社团负责人功能 社团管理 查看和编辑社团基本信息及介绍,更新社团动态。 管理社团内的成员,包括添加、移除和调整角色。 活动策划 创建新活动,设置活动细节(时间、地点、内容、报名方式)。 发布活动通知,通过系统推送给社团成员。 报名管理 查看活动报名情况,管理参与人员名单。 发送活动提醒与通知,确保会员及时了解活动信息。 沟通交流 使用内部消息系统与社团成员进行沟通,发布公告和通知。 建立论坛或讨论区,促进
低通滤波器 滤波算法 滤波
基于滑膜控制smc的3辆协同自适应巡航控制,上层滑膜控制器产生期望加速度,下层通过油门和刹车控制车速,实现自适应巡航控制。 个人觉得从结果图中看出基于滑膜控制的效果非常好,不亚于模型预测控制mpc 并且在实车试验很方便。 文件包含acc巡航建模资料和滑膜控制的资料,还有详细教你运行仿真的步骤,非常的详细,比一般只给文件仿真详细多啦,还有我本人滑膜控制的总结,对于滑膜控制的学习很有帮助。
foc滑膜观测器(SMO+PLL)matlab模型,仿真里面是直接0速闭环启动的效果,当然这是仿真,应用到硬件肯定要加开环启动,目前已经在M4的硬件中实现了,效果还不错,现在出这个模型,matlab 的版本是2021b
springboot158基于springboot的医院资源管理系统,含有完整的源码和报告文档