基于RTMP实现Linux|麒麟操作系统下屏幕|系统声音采集推送

简介: Windows操作系统自问世以来,以其简单易用的图形化界面操作受到大众追捧,为计算机的普及、科技的发展做出了不可磨灭的功绩,也慢慢的成为人们最依赖的操作系统。在中国,90%以上的办公环境都是Windows,学校和各种培训班的培训内容也都是Windows,Windows操作系统已经渗入到各行各业,人们已经习惯了Windows的界面和操作习惯,IT计算机和Windows已经被习以为常的画上了等号。

背景

Windows操作系统自问世以来,以其简单易用的图形化界面操作受到大众追捧,为计算机的普及、科技的发展做出了不可磨灭的功绩,也慢慢的成为人们最依赖的操作系统。在中国,90%以上的办公环境都是Windows,学校和各种培训班的培训内容也都是Windows,Windows操作系统已经渗入到各行各业,人们已经习惯了Windows的界面和操作习惯,IT计算机和Windows已经被习以为常的画上了等号。


但是,我们使用的软件真的安全吗?黑屏事件和棱镜门事件让很多乐观看待或尚未意识到信息安全问题的人们警醒,我们所使用的国外软件并不是安全的,我们的数据完全掌握在别人手中。随着信息安全上升到了国家战略的高度,推行自主可控的国产操作系统势在必行。国产操作系统作为自主可控的基础,市场迅速升温,受到了社会各界的高度关注。


我们有别的选择吗?


其实基于开源软件Linux二次开发的操作系统,近年来的发展趋势非常迅猛。Linux已经有20年历史,尤其近十年经过突飞猛进的发展,Linux桌面操作系统已经远远摆脱了“具备与主流桌面操作系统的可比性”阶段,基于拥有众多优秀的开源应用软件的基础,在软件多样性、硬件兼容性、用户体验等各方面做了大量的改进,现在已经可以满足日常办公的需求。在欧美,我们不时听到一些政府部门将采用Linux桌面办公:慕尼黑市政府用十年的时间,成功的“赶走”了微软;伯明翰市政府、法国国会、瑞士、挪威和南非政府部门也都采用了Linux桌面办公。

国产操作系统|Linux下RTMP同屏推送

在发布国产操作系统|Linux平台的RTMP直播推送SDK之前,大牛直播SDK(官方)的RTMP推送模块已稳定运行在Windows、Android和iOS平台几年了。


相对Windows、Android和iOS平台,Linux在桌面采集等方面,资料非常少,数据采集可以采用调用XLib相关接口实现,本Demo实现的是Linux上实现桌面和系统声音采集,然后使用RTMP协议推出去的一个SDK. 集成调用非常简单。

相关实现

int main(int argc, char *argv[])
{
  signal(SIGINT, &OnSigIntHandler);
  //printf("sizeof(NT_SmartPublisherSDKAPI)=%d\n", sizeof(NT_SmartPublisherSDKAPI));
  LogInit();
  NT_SmartPublisherSDKAPI push_api;
  if (!PushSDKInit(push_api))
  {
    return 0;
  }
  auto push_handle = StartPush(&push_api, "rtmp://192.168.0.154:1935/live/test1", 30);
  if (!push_handle)
  {
    fprintf(stderr, "start push failed.\n");
    push_api.UnInit();
    return 0;
  }
  while (!g_is_exit)
  {
    sleep(2);
  }
  fprintf(stdout, "Skip run loop, is_exit:%d\n", g_is_exit);
  push_api.StopPublisher(push_handle);
  push_api.Close(push_handle);
  push_handle = nullptr;
  push_api.UnInit();
  fprintf(stdout, "SDK UnInit..\n");
  return 0;
}

相关初始化

  void OnSigIntHandler(int sig)
  {
    if (SIGINT == sig)
    {
      g_is_exit = true;
    }
  }
  void LogInit()
  {
    SmartLogAPI log_api;
    memset(&log_api, 0, sizeof(log_api));
    GetSmartLogAPI(&log_api);
    log_api.SetLevel(SL_INFO_LEVEL);
    log_api.SetPath((NT_PVOID)"./");
  }
  bool PushSDKInit(NT_SmartPublisherSDKAPI& push_api)
  {
    memset(&push_api, 0, sizeof(push_api));
    NT_GetSmartPublisherSDKAPI(&push_api);
    auto ret = push_api.Init(0, nullptr);
    if (NT_ERC_OK != ret)
    {
      fprintf(stderr, "push_api.Init failed!\n");
      return false;
    }
    else
    {
      fprintf(stdout, "push_api.Init ok!\n");
    }
    return true;
  }


推送接口封装

  NT_HANDLE StartPush(NT_SmartPublisherSDKAPI* push_api, const std::string& rtmp_url, int dst_fps)
  {
    NT_INT32 pulse_device_number = 0;
    if (NT_ERC_OK == push_api->GetAuidoInputDeviceNumber(2, &pulse_device_number))
    {
      fprintf(stdout, "Pulse device num:%d\n", pulse_device_number);
      char device_name[512];
      for (auto i = 0; i < pulse_device_number; ++i)
      {
        if (NT_ERC_OK == push_api->GetAuidoInputDeviceName(2, i, device_name, 512))
        {
          fprintf(stdout, "index:%d name:%s\n", i, device_name);
        }
      }
    }
    NT_INT32 alsa_device_number = 0;
    if (pulse_device_number < 1)
    {
      if (NT_ERC_OK == push_api->GetAuidoInputDeviceNumber(1, &alsa_device_number))
      {
        fprintf(stdout, "Alsa device num:%d\n", alsa_device_number);
        char device_name[512];
        for (auto i = 0; i < alsa_device_number; ++i)
        {
          if (NT_ERC_OK == push_api->GetAuidoInputDeviceName(1, i, device_name, 512))
          {
            fprintf(stdout, "index:%d name:%s\n", i, device_name);
          }
        }
      }
    }
    NT_INT32 capture_speaker_flag = 0;
    if ( NT_ERC_OK == push_api->IsCanCaptureSpeaker(2, &capture_speaker_flag) )
    {
      if (capture_speaker_flag)
        fprintf(stdout, "Support speaker capture\n");
      else
        fprintf(stdout, "UnSupport speaker capture\n");
    }
    NT_INT32 is_support_window_capture = 0;
    if (NT_ERC_OK == push_api->IsCaptureWindowSupported(NULL, &is_support_window_capture))
    {
      if (is_support_window_capture)
        fprintf(stdout, "Support window capture\n");
      else
        fprintf(stdout, "UnSupport window capture\n");
    }
    NT_HANDLE push_handle = nullptr;
    // if (NT_ERC_OK != push_api->Open(&push_handle, NT_PB_E_VIDEO_OPTION_LAYER, NT_PB_E_AUDIO_OPTION_CAPTURE_SPEAKER, 0, NULL))
    if (NT_ERC_OK != push_api->Open(&push_handle, NT_PB_E_VIDEO_OPTION_SCREEN, NT_PB_E_AUDIO_OPTION_CAPTURE_SPEAKER, 0, NULL))
    {
      return nullptr;
    }
    //push_api->SetXDisplayName(push_handle, ":0");
    //push_api->SetXDisplayName(push_handle, NULL);
    // 视频层配置方式
    //std::vector<std::shared_ptr<nt_pb_sdk::layer_conf_wrapper_base> > layer_confs;
    //auto index = 0;
     第0层填充RGBA矩形, 目的是保证帧率, 颜色就填充全黑
    //auto rgba_layer_c0 = std::make_shared<nt_pb_sdk::RGBARectangleLayerConfigWrapper>(index++, true, 0, 0, 1280, 720);
    //rgba_layer_c0->conf_.red_ = 0;
    //rgba_layer_c0->conf_.green_ = 0;
    //rgba_layer_c0->conf_.blue_ = 0;
    //rgba_layer_c0->conf_.alpha_ = 255;
    //layer_confs.push_back(rgba_layer_c0);
     第一层为桌面层
    //auto screen_layer_c1 = std::make_shared<nt_pb_sdk::ScreenLayerConfigWrapper>(index++, true, 0, 0, 1280, 720);
    //
    //screen_layer_c1->conf_.scale_filter_mode_ = 3;
    //layer_confs.push_back(screen_layer_c1);
    //std::vector<const NT_PB_LayerBaseConfig* > layer_base_confs;
    //for (const auto& i : layer_confs)
    //{
    //  layer_base_confs.push_back(i->getBase());
    //}
    //if (NT_ERC_OK != push_api->SetLayersConfig(push_handle, 0, layer_base_confs.data(),
    //  layer_base_confs.size(), 0, nullptr))
    //{
    //  push_api->Close(push_handle);
    //  push_handle = nullptr;
    //  return nullptr;
    //}
    // push_api->SetScreenClip(push_handle, 0, 0, 1280, 720);
    push_api->SetFrameRate(push_handle, dst_fps); // 帧率设置
    push_api->SetVideoBitRate(push_handle, 2000);  // 平均码率2000kbps
    push_api->SetVideoQualityV2(push_handle, 26); 
    push_api->SetVideoMaxBitRate(push_handle, 4000); // 最大码率4000kbps
    push_api->SetVideoKeyFrameInterval(push_handle, dst_fps*2); // 关键帧间隔
    push_api->SetVideoEncoderProfile(push_handle, 3); // h264 baseline
    push_api->SetVideoEncoderSpeed(push_handle, 3); // 编码速度设置到3
    if (pulse_device_number > 0)
    {
      push_api->SetAudioInputLayer(push_handle, 2);
      push_api->SetAuidoInputDeviceId(push_handle, 0);
    }
    else if (alsa_device_number > 0)
    {
      push_api->SetAudioInputLayer(push_handle, 1);
      push_api->SetAuidoInputDeviceId(push_handle, 0);
    }
    // 音频配置
    push_api->SetPublisherAudioCodecType(push_handle, 1);
    //push_api->SetMute(push_handle, 1);
    if ( NT_ERC_OK != push_api->SetURL(push_handle, rtmp_url.c_str(), NULL) )
    {
      push_api->Close(push_handle);
      push_handle = nullptr;
      return nullptr;
    }
    if ( NT_ERC_OK != push_api->StartPublisher(push_handle, NULL) )
    {
      push_api->Close(push_handle);
      push_handle = nullptr;
      return nullptr;
    }
    return push_handle;
  }


相关界面

20210722122604812.png

总结

实际测试下来,以RTMP推送和RTMP播放为例,整体测试时延都在毫秒级,可满足像内网无纸化、教育同屏等场景需求。


Linux的RTMP推送接口,和Windows平台的相差无几,通用接口四个平台几乎一致,不存在集成难度。


国产操作系统替代windows,也许并不像你想象的那么遥远,使用习惯上也并不像想象中那么难,相信在不久的将来,国产操作系统会真正被市场广泛使用。


国产操作系统下的应用生态的创建,需要国人一点点建立。

相关实践学习
CentOS 7迁移Anolis OS 7
龙蜥操作系统Anolis OS的体验。Anolis OS 7生态上和依赖管理上保持跟CentOS 7.x兼容,一键式迁移脚本centos2anolis.py。本文为您介绍如何通过AOMS迁移工具实现CentOS 7.x到Anolis OS 7的迁移。
相关文章
|
10天前
|
缓存 资源调度 安全
深入探索Linux操作系统的心脏——内核配置与优化####
本文作为一篇技术性深度解析文章,旨在引领读者踏上一场揭秘Linux内核配置与优化的奇妙之旅。不同于传统的摘要概述,本文将以实战为导向,直接跳入核心内容,探讨如何通过精细调整内核参数来提升系统性能、增强安全性及实现资源高效利用。从基础概念到高级技巧,逐步揭示那些隐藏在命令行背后的强大功能,为系统管理员和高级用户打开一扇通往极致性能与定制化体验的大门。 --- ###
34 9
|
8天前
|
人工智能 Android开发 数据安全/隐私保护
移动应用与系统:探索开发趋势与操作系统的协同进化####
当今时代,移动应用不再仅仅是简单的软件工具,它们已成为扩展智能手机及平板等设备功能的关键。本文旨在深入分析当前移动应用的开发趋势,探讨移动操作系统的最新进展及其对应用开发的影响,并阐述两者如何相互促进、协同进化,共同推动移动互联网技术向前发展。 ####
|
10天前
|
移动开发 人工智能 Android开发
移动应用与系统:探索移动开发与操作系统的协同进化####
当今数字化时代,移动设备已成为日常生活不可或缺的一部分。本文旨在深入探讨移动应用开发与移动操作系统之间的紧密关系及其相互影响,揭示技术创新如何推动这一领域的发展。通过分析当前主流移动操作系统的特点、移动应用的开发趋势以及两者间的互动机制,本文为开发者和用户提供了一个全面了解该领域的窗口。 ####
|
7天前
|
开发工具 Android开发 iOS开发
移动应用与系统:涵盖移动应用开发、移动操作系统等相关话题####
本文深入探讨了移动应用开发和移动操作系统的复杂世界。从移动应用开发的基本概念到移动操作系统的核心功能,再到两者如何相互作用以提供无缝的用户体验,本文全面涵盖了这一领域的各个方面。无论你是开发者、技术爱好者还是普通用户,这篇文章都将为你提供有价值的见解。 ####
17 1
|
9天前
|
缓存 运维 网络协议
深入Linux内核架构:操作系统的核心奥秘
深入Linux内核架构:操作系统的核心奥秘
27 2
|
13天前
|
人工智能 物联网 Android开发
移动应用与系统:探索开发趋势与操作系统的协同进化####
本文深入探讨了移动应用开发的当前趋势,以及这些趋势如何与移动操作系统的发展相互影响、协同进化。通过分析最新的技术动态、市场数据及用户行为变化,本文旨在为开发者提供关于未来移动应用开发方向的洞察,并讨论操作系统层面的创新如何促进或制约应用的发展。 ####
|
13天前
|
缓存 网络协议 Linux
Linux操作系统内核
Linux操作系统内核 1、进程管理: 进程调度 进程创建与销毁 进程间通信 2、内存管理: 内存分配与回收 虚拟内存管理 缓存管理 3、驱动管理: 设备驱动程序接口 硬件抽象层 中断处理 4、文件和网络管理: 文件系统管理 网络协议栈 网络安全及防火墙管理
36 4
|
12天前
|
安全 网络协议 Linux
Linux操作系统的内核升级与优化策略####
【10月更文挑战第29天】 本文深入探讨了Linux操作系统内核升级的重要性,并详细阐述了一系列优化策略,旨在帮助系统管理员和高级用户提升系统的稳定性、安全性和性能。通过实际案例分析,我们展示了如何安全有效地进行内核升级,以及如何利用调优技术充分发挥Linux系统的潜力。 ####
33 1
|
14天前
|
安全 Android开发 iOS开发
移动应用与系统:探索移动应用开发与操作系统的协同进化###
【10月更文挑战第29天】 本文深入探讨了移动应用开发与移动操作系统之间的紧密联系与相互促进作用,分析了当前主流移动操作系统(如iOS、Android)的最新特性及其对应用开发的影响,并展望了未来移动应用与系统协同发展的新趋势。通过具体案例分析,揭示了开发者如何利用系统特性优化应用性能,提升用户体验,同时指出了跨平台开发工具的兴起如何进一步模糊了应用与系统间的界限,推动了整个移动互联网生态系统的繁荣发展。 ###
37 2
|
15天前
|
物联网 Linux 云计算
Linux操作系统的演变与未来趋势####
【10月更文挑战第29天】 本文深入探讨了Linux操作系统从诞生至今的发展历程,分析了其在服务器、桌面及嵌入式系统领域的应用现状,并展望了云计算、物联网时代下Linux的未来趋势。通过回顾历史、剖析现状、预测未来,本文旨在为读者提供一个全面而深入的视角,以理解Linux在当今技术生态中的重要地位及其发展潜力。 ####

热门文章

最新文章

下一篇
无影云桌面