转:http://blog.chinaunix.net/uid-23782786-id-3839602.html

前言:
        最近搞这玩样,真是折腾,网上的资料都是片段,而且很少。折腾了4、5还没明白其整个的工作流程。眼看时间一天天地流失,内心哪个捉急啊!

于是,豁出去了,从Makefile开始看。并且在源码中添加调用栈打印,终于搞明白了流程;
        现摘记出来供后来人参考。
----------------------------------------------------------------------------------------------------------------------------------------------------------------------------
前提: 
     芯片:bcm56142 
      
     编译:(kernel model) 
          1. cd system/linux/kernel/x86-generic-2_6/ 
          2. make 
           
          程序的连接使用的是 -static 选项,所以 nsl,m,c,pthread等库只能使用静态库(*.a),否则链接过程会提示找不到库 :
                   /usr/bin/ld: cannot find -lnsl 之类的错误。
           
      编译完成后,进行安装; 
          insmod   linux-kernel-bde.ko 
          insmod   linux-uk-proxy.ko 
          insmod   linux-bcm-diag-full.ko 
          mknod    /dev/linux-uk-proxy c 125 0 
           
1. BCM配置命令操作? 
          我们在shell中执行  ./bcm.user.proxy,就启动了BCM的命令行用户界面。
                这时候,用户就可以使用命令了,比如配置Vlan,端口使能等。 
           
2. 当在开发新产品时,该怎么使用这个SDK呢? 
    当然了,如果不使用这个SDK也是可以的,直接操作设备文件,但是这个是有工作量,需要花时间的... 
     
    我们在执行了 ./bcm.user.proxy时,实际启动的是一个用户空间程序,
           对应文件 systems/linux/kernel/user/bcm-diag-proxy/bcm-diag-proxy.c里的main()函数。

main() 
    ( 
        1. 首先打开设备文件 /dev/linux-uk-proxy。 
        2. 启动一个读线程,用于从内核 _service_list(name "BCM DIAG SHELL")的sndqueue发送队列
           读取内核执行命令后的输出信息。 
        3. 当前进程则将用户输入定向到标准输入,如果用户输入了字符,则将其发送到
            内核_service_list(name "BCM DIAG SHELL")rcvqueue 接收队列
    )

当我们在 BCM> 提示符下输入一条命令,比如 show pci 
             命令"show pci"被传送到了内核_service_list(name "BCM DIAG SHELL")rcvqueue接收队列上。

那么show pci命令怎么被执行了呢? 
     
    回到前面的 insmod linux-bcm-diag-full.ko 
    当这个模块被插入内核时,其初始化函数会调用 _init() 
    /* systems/linux/kernel/modules/shared/gmodule.c */ 
    int __init init_module()   //内核模块加载函数,即insmod时会调用的函数,module_init()最终也是掉这个函数 
    {     
            _gmodule = gmodule_get(); 
        ...... 
        if(_gmodule->init) 
        { 
              if((rc = _gmodule->init()) < 0) //调用 _init() 
              { 
              } 
        } 
    } 
     
    /* systems/linux/kernel/modules/bcm-diag-full/bcm-diag-full.c */ 
    _init() 
    { 
        最重要的活就是创建了一个处理shell命令的内核线程:_bcm_shell() 
    } 
     
    /* systems/linux/kernel/modules/bcm-diag-full/bcm-diag-full.c */ 
    _bcm_shell() 
    { 
        调用函数 diag_shell(),这个就是最重要的函数了。 
    } 
     
    /* src/appl/diag/system.c */ 
    diag_shell() 
    { 
        diag_init();    // 这个没啥好说的,自己看代码,初始化了一些链表,并打印BCM相关信息 
        sysconf_init();  
        sysconf_probe();//这里去探测系统中存在的BCM家族的芯片,主要是通过读PCI配置空间的ID 
        sysconf_attach(); //这里就是将探测到的设备对应的函数安装到对应的钩子上 
        while(1) 
        { 
            sh_process(-1, "BCM", TRUE);  //这个东东就一直轮询用户的输入,并执行用户的命令了。 
        } 
    } 
     
    /* src/appl/diag/shell.c */ 
    sh_process() 
    { 
        while(1) 
        { 
            sal_readline(); //关注一下这个函数,这个函数就是
                                  //             去从内核_service_list(name "BCM DIAG SHELL")rcvqueue接收队列上取用户的命令 
        } 
         
      ...... 
      if( cur_mode == BCMX_CMD_MODE ) 
          sh_process_command();     //看到没,这时候才是真正的去执行用户输入的命令了。 
    } 
     
   /* src/sal/appl/io.c */ 
   sal_readline() 
   { 
           sal_console_gets();     
   } 
    
   /* src/sal/appl/linux/console.c */ 
   sal_console_gets() 
   { 
           tty_gets(); 
   } 
    
   /* systems/linux/kernel/modules/bcm-diag-full/bcm-diag-full.c */ 
   tty_gets() 
   { 
           while(!(rc >= 0)) 
           { 
                 rc=linux_uk_proxy_recv();    //直接从内核_service_list(name "BCM DIAG SHELL")rcvqueue队列上
                                                             // 取用户的命令 
           } 
   } 
    
    
   /* systems/linux/kernel/modules/uk-proxy/linux-uk-proxy.c */ 
   linux_uk_proxy_recv() 
   {     
       return _list_op(&s->recv_q.data, &s->recv_q.free, data, len, _LIST_OP_F_GET | _LIST_OP_F_KERN); 
   } 
    
   从以上分析可以看出,linux-bcm-diag-full.ko 是很重要的模块,用于接口用户的,执行bcm 命令。 
     
写这篇文档,一是记录下,日后自己再做BCM的开发,可以省时省力些;二来给像我一样刚接触BCM开发的后来者以参考,少走弯路。 
当然,要向了解所有的,最好还是看下他的makefile工程,虽然可能会花上一两天时间,但是对你看代码很有帮助的, 
因为代码里好多一样的文件名和函数名,不看Makefile你不知道调哪个了. 
 
对于开发BCM应用: 
1. 你可以修改 bcm-diag-proxy.c的main线程,将其的输入重定向一个文件,然后你懂的... 
2. 直接操作 /dev/linux-uk-proxy文件(open,ioctl,对应内核的模块linux-uk-proxy.c 中的_ioctl)

 

BCM 交换机开发的更多相关文章

  1. 快速掌握RabbitMQ(二)——四种Exchange介绍及代码演示

    在上一篇的最后,编写了一个C#驱动RabbitMQ的简单栗子,了解了C#驱动RabbitMQ的基本用法.本章介绍RabbitMQ的四种Exchange及各种Exchange的使用场景. 1 direc ...

  2. PISCES: A Programmable, Protocol-Independent Software Switch

    Name of article:PISCES: A Programmable, Protocol-Independent Software Switch Origin of the article:S ...

  3. H3C S6800交换机 BCM shell命令

    H3C S6800交换机 BCM shell命令 http://wgli978.blog.163.com/blog/static/13592877220172315858831/ <H3C> ...

  4. Ryu控制器编程开发——packet_in和packet_out简易交换机实现

    Ryu控制器二次开发,实现一个简单的只能够简单地广播数据包的交换机. from ryu.base import app_manager from ryu.controller import ofp_e ...

  5. 开发人员必读openstack网络基础2:交换机、路由器、DHCP

    我们在使用openstack的过程中,会遇到创建虚拟机路由器.交换机等,那么1.他们的作用到底是什么?2.DHCP为什么会产生,它的作用是什么? 个人总结:交换机:一般用在同一网段,工作在数据链路层, ...

  6. maven实战(01)_搭建开发环境

    一 下载maven 在maven官网上可下载maven:http://maven.apache.org/download.cgi 下载好后,解压.我的解压到了:D:\maven\apache-mave ...

  7. 开发人员必读openstack网络基础

    云计算中的网络非常复杂,需要对网络的基础理论有一定的认识和了解,转载网上针对openstack中涉及到网络概念的文章 开发人员必读openstack网络基础1:什么是L2.L3 开发人员必读opens ...

  8. nagios二次开发(四)---nagios监控原理和nagios架构简介

    nagios监控原理 下面根据上面摘自网络的原理图对nagios的监控原理进行一下简单的说明: 1.nagios通过nsca进行被动监控.那么什么是被动监控呢?被动监测:就是指由被监测的服务器主动上传 ...

  9. 【WP 8.1开发】推送通知测试服务端程序

    所谓推送通知,用老爷爷都能听懂的话说,就是: 1.我的服务器将通知内容发送到微软的通知服务器,再由通知服务器帮我转发消息. 2.那么,微软的推送服务器是如何知道我的服务器要发消息给哪台手机呢?手机客户 ...

随机推荐

  1. eclipse JavaEE版"javax.servlet.http.HttpServlet" was not found on the Java Build Path问题的解决办法

    使用eclipse JavaEE 版,新建 Dynamic Web Project 项目.在项目里添加 JSP 文件,会在文件头部出现错误提示.提示语句为:The superclass "j ...

  2. 在 RHEL/CentOS 7 上配置NTP时间服务器

    一.NTP简介 网络时间协议 - NTP - 是运行在传输层 123 号端口的 UDP 协议,它允许计算机通过网络同步准确时间.随着时间的流逝,计算机内部时间会出现漂移,这会导致时间不一致问题,尤其是 ...

  3. Struts2 主题和模板

    实际本章教程开始之前,让我们看看由http://struts.apache.org给出的几个定义: Term 描述 tag A small piece of code executed from wi ...

  4. Fibonacci series(斐波纳契数列)的几种常见实现方式

    费波那契数列的定义: 费波那契数列(意大利语:Successione di Fibonacci),又译费波拿契数.斐波那契数列.斐波那契数列.黄金切割数列. 在数学上,费波那契数列是以递归的方法来定义 ...

  5. asp.net C#实现下载文件的六种方法实例

    protected void Button1_Click(object sender, EventArgs e)  {  /*  微软为Response对象提供了一个新的方法TransmitFile来 ...

  6. python3----基础 用while循环+iter()+next() 实现对字符串的遍历与输出

    my_str = 'hello' # for循环 for v in my_str: print(v) # while 配合迭代器实现字符串的遍历 ite = iter(my_str) while Tr ...

  7. ubuntu虚拟化平台libvrit-bin

    http://download.cirros-cloud.net/0.3.5/ 下载 cirros-0.3.5-x86_64-disk.img 因为 KVM(准确说是 Libvirt)默认不接受远程管 ...

  8. boost::lockfree::stack

    #include <boost/thread/thread.hpp> #include <boost/lockfree/stack.hpp> #include <iost ...

  9. 【BZOJ3994】[SDOI2015]约数个数和 莫比乌斯反演

    [BZOJ3994][SDOI2015]约数个数和 Description  设d(x)为x的约数个数,给定N.M,求   Input 输入文件包含多组测试数据. 第一行,一个整数T,表示测试数据的组 ...

  10. 剖析与优化 Go 的 web 应用

    https://mp.weixin.qq.com/s/HDsbZLOK3h8-XjejvPH2sA https://studygolang.com/articles/12685