PeerCast 分析报告

原创 2006年06月05日 23:46:00
PeerCast 分析报告
目录
/------------------------------------------------------------------------
1. 程序基本结构
1.1 Simpel工程分析
    1.1.1 Simple工程的主要功能函数模块
    1.1.2 WinMain函数分析
    1.1.3 WndProc函数分析
    1.1.4 ChanInfoProc函数分析
2. Corelib内部分析
3.主程序说明
 
/------------------------------------------------------------------------
函数名
类或者结构体
注释
变量或者对象
关键字
//---------------------------代表程序段开始处
//***************************代表程序段结束处
1.程序基本结构
 
本报告分析方法是从PeerCast的主程序开始,然后跟进到LIB函数说明其功能。PeerCast是一个支持多平台SDK开发包,因此源码里可以看到Win32和Unix两种版本。
整个WorkSpace(工作区)包含3个Projects(工程),包括corelib、Simple、winamp2。
 
u       PeerCast的主要功能都由Corelib实现,它通过peercast.h导出两个类PeercastInstance和PeercastApplication
u       winamp2是程序响应后调用的播放插件
u       Simple工程是程序的主体实现,通过继承peercast.h导出的两个类实现程序功能
   
1.1 Simple工程分析
 
   1.1.1 Simple工程的主要功能函数模块:
 
//注册程序窗体
ATOMMyRegisterClass(HINSTANCE hInstance);
//初始化程序窗体
BOOL InitInstance(HINSTANCE, int);
//负责传递消息到程序窗口
LRESULT CALLBACK  WndProc(HWND, UINT, WPARAM, LPARAM);
//负责传递信息到接收频道后的频道信息窗口
LRESULT CALLBACK      ChanInfoProc(HWND, UINT, WPARAM, LPARAM);
//显示任务栏图标
void setTrayIcon(int type, const char *,const char *,bool);
//显示频道信息窗口信息
void flipNotifyPopup(int id, ServMgr::NOTIFY_TYPE nt);
 
1.1.2 WinMain函数分析
 
程序的入口函数是 int  APIENTRY  WinMain(HINSTANCE hInstance,
                                               HINSTANCE  hPrevInstance,
                                               LPSTR      lpCmdLine,
                                               int           nCmdShow)
     其中第一个参数是程序实例句柄,第二个参数以被废弃,第四个参数是
InitInstance函数中用到的参数。第三个参数很重要,它是网页与程序交互的桥梁。
 
WinMain函数中lpCmdLine有五种可能值:-inifile、-zen、-multi、-kill、-url。
我们重点分析其值为-url的情况,当用户点击网页播放链接时,程序通过以下代码去掉了连接的协议头peercast://pls,得到了标准的media player播放文件名chanURL:
//-------------------------------------------------------
if (strnicmp(tmpURL,"peercast://",11)==0)
 {
         if (strnicmp(tmpURL+11,"pls/",4)==0)
                chanURL = tmpURL+11+4;
         else
                chanURL = tmpURL+11;
         showGUI = false;
 }
    //***************************************
 
     WinMain初始化了corelib的两个导出类的继承类对象:
    //----------------------------------------------------------
peercastInst = new MyPeercastInst();
        peercastApp = new MyPeercastApp();
//***************************************
 
下面的函数生成了新的ServMgr类和ChanMgr类对象,根据默认设置生成INI文件,生成了INI中的 SessionID和BroadcastID.。主要功能在ServMgr类的loadSettings函数实现。
//----------------------------------------------------------
peercastInst->Init();
//***************************************
 
下面的函数判断chanURL是否存在,也就是判断用户是否有点击链接
//----------------------------------------------------------
if (chanURL)
     {
            ChanInfo info;
            servMgr->procConnectArgs(chanURL,info);
            chanMgr->findAndPlayChannel(info,false);
     }
//***************************************
 
procConnectArgs函数的功能是:
l         sip - add network connection to client with channel
l         pip - add private network connection to client with channel
l         ip - add hit
l         tip - add tracker hit
findAndPlayChannel函数的功能是通过ChanInfo播放频道文件,主要功能在Channel类里的findAndPlayChannelProc函数实现。
 
//----------------------------------------------------------
peercastInst->saveSettings();
    peercastInst->quit();
//***************************************
saveSettings函数设置并保存INI文件,主要功能在ServMgr类的saveSettings函数实现。
 
1.1.3 WndProc函数分析
      //-----------------------------------------------------------
LRESULT CALLBACK WndProc(HWND hWnd, UINT message, WPARAM wParam, LPARAM lParam)
      WndProc主要是响应自定义消息和系统消息。
      Message的可能值:
                      WM_SHOWGUI             UI界面初始化
                      WM_TRAYICON            发送左、右键点击下标消息
                      WM_COPYDATA            Muti情况下播放
                WM_GETPORTNUMBER     得到端口号
                WM_SHOWMENU           检查PC新版本,响应左键、右键弹出菜单,判断是否有最近频道,有则加入到列表
                WM_CREATE               同WM_CREATE
                WM_COMMAND            主要判断wParam底四字节情况,响应弹出菜单消息和插入菜单的自定义消息(点击连接后增加菜单)
 WM_DESTROY:              系统退出
 
1.1.4 ChanInfoProc函数分析
 
ChanInfoProc函数主要在响应WM_COMMAND的四个自定义消息中的INFO_CMD消息,由DialogBox函数调用。
   
 
 
2. corelib内部分析
 
2.1 corlib内部各类功能调试心得
 
mms.cpp、 mp3.cpp、 nsv.cpp ogg.cpp都是读指定文件格式的读包协议,
jis.cpp是日文编码。
 
所有客户端必须向server交互自己的消息(通过xml),靠服务端来获得其它客
户的地址后在通过servhs类向其它客户握手,通过gnutella里的gnupacket
通信。
 
settings.html页,save按钮响应ServMgr::saveSettings函数。利用
inifile.cpp里的inifile类重写ini文件。inifile类封装了stream.cpp里的stream类,stream类主要通过CFile的fopen()和fread、fwrite、fseek打开、读写ini文件 。本质就是通过CFile类读写文件。
 
         brocast.html页,Create Relay按钮响应url.cpp中的stream函数,这个函数调用streamURL函数提交播放地址,并把地址按照一定规则转换,得到ID。
 
         Peercast右键菜单的advanced里通过showHTML函数打开explorer,具体过程为: showHTML----à sys->callLocalURL ----à ShellExecute(mainWindow, NULL, cmd, NULL, NULL, SW_SHOWNORMAL);
  
         当点击YP上的连接后,响应channel.cpp里的ChanHitList类的构造函数,接着响应Channnel类的构造函数。PC会在左键菜单加入频道,频道包含play和info选项,点击play按钮,PC响应
sys->executeFile(fname)---à ShellExecute(NULL,"open",fname,NULL,NULL,SW_SHOWNORMAL)
这时的fname为play.pls。
 
如果直接在网页里输入网址,会响应程序的wm_copydata消息,channel.cpp
中函数响应过程:                                                     
findandPlayChannel--àfindAndPlayChannelProc--àfindChannelByNameID--àfindAndRelay--àfindChannelByNameID---àCreateChannel--àfindChannelByNameID---àwhile(ch)这时候有值了---àreturn ch;得到channel了。
 
 
servMgr::start是程序运行时候就自动启动的
corelib里的构造函数加断点
响应顺序是:
Sys::Sys()
servMgr::servMgr()
ChanMgr::ChanMgr()
ServMgr::Start()
GnuIDList::GunIDList(int max)
然后点击pc的brocast.html链接按钮,响应handshakeHTTP
接下来是HTML::HTML
进入yp页面点击连接后响应
ChanHitList::ChanHitList()
然后是Channel::Channel()
 
servent类执行顺序:
Servent::serverProc
Servent::initIncoming
Servent::incomingProc
Servent::givProc
Servent::handshakeIncoming()
Servent::handshakeHTTP
PCPStream::procAtom
Corlib里的类
 
Xml类里嵌套了Node类。
Url里是URLSource类。
Sys里String类,Radom类,Sys类,WEvent类,Wlock类,Threadinfo类,LogBuffer类
Stream里Stream类,FileStream类,MemoryStream类,IndirectStream类
Stats里Stats类
ServMgr里BCID类,ServHost类,ServFilter类,ServMgr类
Servent里Servent类
Http里HTTPException类,Cookie类,CookieList类HTTP类
Html里Html类
Gnutella里GnuPacket类里嵌套了Hash类,GnuPacketBuffer类,GnuStream类
cstream里ChanPacket类,ChanPacketBuffer类ChannelStream类
Common里各种EXCEPTION,GnuID类,GnuIDList类,
Channel里TrackInfo类,ChanInfo类,ChanHit类,ChanHitList类,ChanHitSearch类
ChanMeta类,RawStream类,PeercastStream类,ChannelSource类,PeercastSource类
Channel类,ChanMgr类PlayList类
Atom里AtomSteam类
 
==================接口=========================
peercast.h 中定义了PeercastInstance是外部应用给corelib的接口,peercast.h中只有定义有部分实现,外部调用者可以实现该接口,通过对在 peercast.cpp中定义的全局变量PeercastInstance* peercastInst重新赋值,来实现这一点.
 
peercast.h 中定义的PeercastApplication *peercastApp是core给应用程序的接口,其中包括增加和减少频道,启动和停止频道.还有一个test方法!待调查是做什么的.
 ==================http server=====================
servhs.cpp中实现了对http的一些指令的处理.每一个http访问都通过handshakeHTTP函数来处理.
handshakeCMD函数包含对各种形式为/admin?cmd=的http请求的处理
发布频道的过程:当发布一个频道的时候当调用请求的处理为/admin?,经过hadshakeHTTP的分派,转而调用 handshakeCMD(对应的cmd=fetch),先根据表单的内容(从url解析出来)填充一个新的ChanInfo,转而调用ChanMgr的 createChannel方法,该方法把新的频道加入到频道链表中,然后调用startUrl方法中启用了一个新的线程,该线程的执行函数是 stream方法,其中的主要的功能是调用了findHitList方法,如果找到那么调用addHitList
 =================core lib========================
ChanMgr 中包含频道的链,其中变量channel是频道链表的表头指针,
 
2.2 corlib总体分析
 
 2.2.1 stream.cpp分析
     
       Stream.cpp有三个类 FileStream String IndirectStream MemoryStream
 
       FileStream封装了File类
 
       FileStream::openReadOnly(const char *fn) 以只读二进制方式打开文件
       FileStream::openWriteReplace(const char *fn) 以可写二进制方式打开文件
       FileStream::openWriteAppend(const char *fn)   以附加二进制方式打开文件
       FileStream::close() 关闭文件
       FileStream::rewind() 让文件指针重新定位到头部
       FileStream::length() 返回文件长度
       FileStream::eof() 判断文件是否结束
       FileStream::read(void *ptr, int len) 在指定的文件指针位置读文件
       FileStream::write(const void *ptr, int len) 在指定的文件指针位置写文件
       FileStream::flush() 清空文件缓冲区
       FileStream::pos() 得到文件指针的位置
       FileStream::seekTo(int pos) 搜索文件到指定位置
     
2.2.2 wsocket.cpp分析
 
WSAClientSocket::init() //建立winsock2.0版本的套接字
ClientSocket::getHostname(char *str,unsigned int ip) //把IP转换为主机名
ClientSocket::getIP(char *name) //同过主机名得到IP
WSAClientSocket::setLinger(int sec) //socket延时等待未到达数据,sec为0时不设置延时
WSAClientSocket::setBlocking(bool yes) //设置socket为阻塞/非阻塞模式
WSAClientSocket::setNagle(bool on) //可以减轻TCP协议在传送小包上的问题.如果没有配置service nagle命令,在远程登录进程钟每个字符都需要占用CPU中断
(感觉层级SOL_SOCKET应该改成IPPROTO_TCP1)   
WSAClientSocket::setReuse(bool yes)// 可以让一个socket绑定正在使用的地址
WSAClientSocket::resolveHost(const char *hostName)// 通过主机名得到hostent结构体
WSAClientSocket::open(Host &rh) //建立TCP socket
WSAClientSocket::checkTimeout(bool r, bool w) //检查连接是否超时。多端口复用函数select在调用前要首先设置监听的端口数目,FD_ZERO是清空端口集,FD_SET    是设置端口集。select函数常常用在用一个进程监听多个服务器端socket。 
WSAClientSocket::getLocalHost() //得到主机IP
WSAClientSocket::connect() //连接远端地址,第一个参数怎么是unsigned int型,不是socket型?
WSAClientSocket::read(void *p, int l) //得到接收字节
WSAClientSocket::readUpto(void *p, int l) //和上面的函数完全一样- -!
WSAClientSocket::write(const void *p, int l) //发送数据
WSAClientSocket::bind(Host &h) //建立TCPsocket设置为非阻塞socket,设置为可重复绑定地址,绑定主机地址,并开始监听。
WSAClientSocket::accept() //接收连接请求,设置为非阻塞模式
WSAClientSocket::close() //关闭socket
WSAClientSocket::readReady() //判断是否在监听多个服务器端socket
 
2.2.3 xml.cpp分析
       
        XML::Node::add(Node *n) //可以看成XML::Node::    add(Node *n)添加兄弟结点或者子结点
       nibsToByte(char n1, char n2) //没用到
       XML::Node::getBinaryContent(void *ptr, int size) //没用到
       XML::Node::setBinaryContent(void *ptr, int size) //没用到
       XML::Node::setContent(const char *n) //复制字符串
       XML::Node::setAttributes(const char *n) //设置属性名和属性值的关联,判断并掉过/和=来读取属性???在XML::Node::构造函数里用
     XML::Node::init() //清空父、兄弟、子结点
      XML::Node::findAttrInt(const char *name) //通过属性名得到属性值,比如findAttrInt("bitrate")得到比特率的数值
      XML::Node::findAttrID(const char *name) //通过属性名得到属性ID
      XML::Node::findAttr(const char *name) //通过查找对比属性名得到属性值的ID
2.2.3 servhs.cpp分析
Servent::handshakeHTTP(HTTP &http, bool isHTTP)
       if (http.isRequest("GET /"))
       else if (http.isRequest("GIV"))
       else if (http.isRequest(PCX_PCP_CONNECT))
       else if (http.isRequest("PEERCAST CONNECT"))
else if (http.isRequest("SOURCE"))
else if (http.isRequest(servMgr->password))

oracle awr报告生成和分析

最近由于数据库cpu占用非常高,导致VCS常常自动切换,引起很多问题。 最近学习一下数据库awr分析数据库sql执行性能的分析报告。下面将初步讲解一下: 1、先登陆数据库,生成awr报告。 li...
  • cuker919
  • cuker919
  • 2013年04月07日 14:20
  • 12389

Jmeter聚合报告分析

聚合报告:Aggregate Report Label:每个JMeter的element的Name值。例如HTTP Request的Name #Samples:发出请求数量。如第三行记录,模拟20个用...
  • w_xuexi666
  • w_xuexi666
  • 2017年03月28日 23:12
  • 1969

【深度长文】循序渐进解读Oracle AWR性能分析报告

【深度长文】循序渐进解读Oracle AWR性能分析报告 原创 2016-10-19 韩锋 DBAplus社群 http://mp.weixin.qq.com/s?__biz=...
  • u010719917
  • u010719917
  • 2016年10月28日 12:43
  • 1179

PeerCast 分析报告

目录 /------------------------------------------------------------------------ 1. 程序基本结构 1.1 Simpel...
  • vbLittleBoy
  • vbLittleBoy
  • 2014年05月12日 12:51
  • 596

数据解决方案:原力大数据教你如何撰写数据分析报告

转载于原力大数据,作者简介江颖,帷策智能、原力大数据创始人兼CEO,大数据应用实施领域资深专家,中国大数据商业实践先行者。武汉大学管理学硕士,13年大数据研究及应用实践经历,曾任中国移动资深大数据分析...
  • sinat_26917383
  • sinat_26917383
  • 2016年12月30日 11:05
  • 1394

《操作系统》期末考试卷面成绩不及格率过高-分析报告和整改措施—张同光

《操作系统》期末考试卷面成绩不及格率过高—分析报告和整改措施   分析报告 2013―2014学年度第1学期,《操作系统》期末试卷A卷。 考试班...
  • ztguang
  • ztguang
  • 2016年02月01日 00:00
  • 931

大数据分析报告显示,多少年薪才能让80后摆脱中年危机?

现如今,80后已成为职场、家庭、社会的中流砥柱。第一批80后已经37岁,最后一批也满了28岁。那么在这一代人中,现在月薪多少才算跑赢同龄人的“人生赢家”呢?而走在奔四路上的80后,成了中年危机的重灾区...
  • BtB5e6Nsu1g511Eg5XEg
  • BtB5e6Nsu1g511Eg5XEg
  • 2017年12月04日 00:00
  • 425

数字解读中国IT行业人群生活质量分析报告

调查背景:   我国的 IT 行业目前已经历了近 30 年的发展,在兼具了规模和影响力的同时也形成了完整的产业链条。随着 IT 产业规模的不断扩大,专业门类的日益齐全,技术水平的显著提升...
  • yuanhanlin1989
  • yuanhanlin1989
  • 2014年02月20日 14:09
  • 708

数据分析报告怎么用

数据分析报告怎么用 本文是作者基于自身多年数据分析的经验,总结的一体化数据分析框架,简单地介绍下数据分析能分析能落地的几点。enjoy~ 大数据,这个被炒烂了的概念,现如今已被人工智能替代...
  • jiahaowanhao
  • jiahaowanhao
  • 2018年01月04日 20:57
  • 85

元征科技Golo3 硬件拆解分析报告——榕树投资研究员.杜志君(2014-07-21

(2014-07-21 14:09:17) 转载▼ 标签:  股票 分类: TMT 硬件拆解分析报告——榕树投资研究员.杜志君" ...
  • mirkerson
  • mirkerson
  • 2014年09月02日 14:52
  • 6999
内容举报
返回顶部
收藏助手
不良信息举报
您举报文章:PeerCast 分析报告
举报原因:
原因补充:

(最多只允许输入30个字)