当前位置: 首页 > news >正文

coreutils5.0 uname命令和源码分析

  uname   /* 默认输出内核名。和uname -s一样 */


  -a, --all  /* 按以下顺序打印所有信息 */
  -s  /* 打印内核名称 */
  -n  /* 打印主机名 */
  -r  /* 打印内核版本发行号 */
  -v  /* 打印内核版本 */
  -m  /* 打印机器硬件名称 */
  -p  /* 打印CPU类型 */
  -i  /* 打印硬件平台 */
  -o   /* 打印操作系统 */

/* uname -- print system information

   Copyright 1989, 1992, 1993, 1996, 1997, 1999, 2000, 2001, 2002 Free
   Software Foundation, Inc.

   This program is free software; you can redistribute it and/or modify
   it under the terms of the GNU General Public License as published by
   the Free Software Foundation; either version 2, or (at your option)
   any later version.

   This program is distributed in the hope that it will be useful,
   but WITHOUT ANY WARRANTY; without even the implied warranty of
   MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
   GNU General Public License for more details.

   You should have received a copy of the GNU General Public License
   along with this program; if not, write to the Free Software Foundation,
   Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA.  */

/* Written by David MacKenzie <djm@gnu.ai.mit.edu> */

#include <config.h>
#include <stdio.h>
#include <sys/types.h>
#include <sys/utsname.h>
#include <getopt.h>

#if HAVE_SYSINFO && HAVE_SYS_SYSTEMINFO_H
# include <sys/systeminfo.h>
#endif

#if HAVE_SYSCTL && HAVE_SYS_SYSCTL_H
# include <sys/param.h> /* needed for OpenBSD 3.0 */
# include <sys/sysctl.h>
# ifdef HW_MODEL
#  ifdef HW_MACHINE_ARCH
/* E.g., FreeBSD 4.5, NetBSD 1.5.2 */
#   define UNAME_HARDWARE_PLATFORM HW_MODEL
#   define UNAME_PROCESSOR HW_MACHINE_ARCH
#  else
/* E.g., OpenBSD 3.0 */
#   define UNAME_PROCESSOR HW_MODEL
#  endif
# endif
#endif

#include "system.h"
#include "error.h"
#include "closeout.h"

/* The official name of this program (e.g., no `g' prefix).  */
#define PROGRAM_NAME "uname"

#define AUTHORS "David MacKenzie"

/* 用于修改toprint的值 */
/* 内核名称 */
#define PRINT_KERNEL_NAME 1

/* 主机名 */
#define PRINT_NODENAME 2

/* 内核发行号 */
#define PRINT_KERNEL_RELEASE 4

/* 内核版本 */
#define PRINT_KERNEL_VERSION 8

/* 硬件名 */
#define PRINT_MACHINE 16

/* CPU类型 */
#define PRINT_PROCESSOR 32

/* 硬件平台  */
#define PRINT_HARDWARE_PLATFORM 64

/* 操作系统  */
#define PRINT_OPERATING_SYSTEM 128

/* 保存程序名 = argv[0] */
char *program_name;

/* 长选项结构体 */
static struct option const long_options[] =
{
  {"all", no_argument, NULL, 'a'}, /* "长选项名称","选项:no_argument代表该长选项无需参数","元素三为NULL时,getopt将直接返回元素四,不为空时参数三将保存元素4" */
  {"kernel-name", no_argument, NULL, 's'},
  {"sysname", no_argument, NULL, 's'},	/* 即将弃用,该源码是5.0,不知道9.0是否以弃用.  */
  {"nodename", no_argument, NULL, 'n'},
  {"kernel-release", no_argument, NULL, 'r'},
  {"release", no_argument, NULL, 'r'},  /* 即将弃用,该源码是5.0,不知道9.0是否以弃用.  */
  {"kernel-version", no_argument, NULL, 'v'},
  {"machine", no_argument, NULL, 'm'},
  {"processor", no_argument, NULL, 'p'},
  {"hardware-platform", no_argument, NULL, 'i'},
  {"operating-system", no_argument, NULL, 'o'},
  {GETOPT_HELP_OPTION_DECL},
  {GETOPT_VERSION_OPTION_DECL},
  {NULL, 0, NULL, 0} /* 没有参数时返回0 */
};

/* 传入的命令行参数的错误函数,传入不支持的参数就调用该函数 */
void
usage (int status)
{
  if (status != 0)
    fprintf (stderr, _("Try `%s --help' for more information.\n"),
	     program_name);
  else
    {
      /* 调用错误函数时,提示该命令用法,命令没有参数时将和-s一样 */
      printf (_("Usage: %s [OPTION]...\n"), program_name);
      fputs (_("\
Print certain system information.  With no OPTION, same as -s.\n\
\n\
  /* 按以下顺序打印所有信息 */
  -a, --all                print all information, in the following order:\n\
  /* 打印内核名称 */
  -s, --kernel-name        print the kernel name\n\
  /* 打印主机名 */
  -n, --nodename           print the network node hostname\n\
  /* 打印内核版本发行号 */
  -r, --kernel-release     print the kernel release\n\
"), stdout);
      fputs (_("\
  /* 打印内核版本 */
  -v, --kernel-version     print the kernel version\n\
  /* 打印机器硬件名称 */
  -m, --machine            print the machine hardware name\n\
  /* 打印CPU类型 */
  -p, --processor          print the processor type\n\
  /* 打印硬件平台 */
  -i, --hardware-platform  print the hardware platform\n\
  /* 打印操作系统 */
  -o, --operating-system   print the operating system\n\
"), stdout);
      fputs (HELP_OPTION_DESCRIPTION, stdout);
      fputs (VERSION_OPTION_DESCRIPTION, stdout);
      printf (_("\nReport bugs to <%s>.\n"), PACKAGE_BUGREPORT);
    }
  exit (status);
}

/* Print ELEMENT, preceded by a space if something has already been
   printed.  */

static void
print_element (char const *element)
{
  static int printed;
  if (printed++)
    putchar (' ');
  fputs (element, stdout);
}

int
main (int argc, char **argv)
{
  int c;
  static char const unknown[] = "unknown";

  /* Mask indicating which elements to print. */
  /* 指示要打印哪些元素的掩码(最后用该值判断用户需要输出什么信息) */
  unsigned toprint = 0;

  /* 程序初始化 */
  program_name = argv[0];  /* 程序名 */
  setlocale (LC_ALL, "");  /* 获取系统编码、地区代码等信息 */
  bindtextdomain (PACKAGE, LOCALEDIR);
  textdomain (PACKAGE);
  
  atexit (close_stdout);/* 出口函数 */

  /* 解析命令行参数 */
  while ((c = getopt_long (argc, argv, "asnrvmpio", long_options, NULL)) != -1) /* c保存用户传入的参数 */
    {
      /* 判断用户传入的函数来修改toprint的值 */
      switch (c)  
	{
	case 0:
	  break;

	case 'a':
	  toprint = -1;
	  break;

   /* 内核名称,值=1 */
	case 's':
	  toprint |= PRINT_KERNEL_NAME;
	  break;

   /* 主机名称,值=2 */
	case 'n':
	  toprint |= PRINT_NODENAME;
	  break;

   /* 内核发行号,值=4 */
	case 'r':
	  toprint |= PRINT_KERNEL_RELEASE;
	  break;

   /* 内核版本,值=8 */
	case 'v':
	  toprint |= PRINT_KERNEL_VERSION;
	  break;

   /* 硬件名,值=16 */
	case 'm':
	  toprint |= PRINT_MACHINE;
	  break;

   /* CPU类型,值=32 */
	case 'p':
	  toprint |= PRINT_PROCESSOR;
	  break;

   /* 硬件平台,值=64 */
	case 'i':
	  toprint |= PRINT_HARDWARE_PLATFORM;
	  break;

   /* 操作系统,值=128 */
	case 'o':
	  toprint |= PRINT_OPERATING_SYSTEM;
	  break;

   /* 在system.h内定义 在这case内会exit退出程序 */
	case_GETOPT_HELP_CHAR;

   /* 在system.h内定义 在这case内会exit退出程序 */
	case_GETOPT_VERSION_CHAR (PROGRAM_NAME, AUTHORS);

	default:
	  usage (EXIT_FAILURE);  /* 如果以上case都没有触发的话,调用usage帮助函数,退出程序,EXIT_FAILURE的值为1,定义在stdlib */
	}
    }

  /* getopt调用的次数不等于main传入的参数就报错。用来测试传入的参数是否全部都getopt成功。因为每次调用一次getopt都会使optind +1 */
  if (optind != argc)
    usage (EXIT_FAILURE);

  /* 不带参数默认转成内核名称 */
  if (toprint == 0)
    toprint = PRINT_KERNEL_NAME;

  /* 参数值==1 | 2 | 4 | 8 | 16 进入该if */
  if (toprint
       & (PRINT_KERNEL_NAME | PRINT_NODENAME | PRINT_KERNEL_RELEASE
	  | PRINT_KERNEL_VERSION | PRINT_MACHINE))
    {
      struct utsname name;  /* 获取系统相关信息的结构体 */

      if (uname (&name) == -1)  /* 填充ustname结构 */
	error (EXIT_FAILURE, errno, _("cannot get system name"));  /* 无法获取系统名称 */

      if (toprint & PRINT_KERNEL_NAME)
	print_element (name.sysname);
      if (toprint & PRINT_NODENAME)
	print_element (name.nodename);
      if (toprint & PRINT_KERNEL_RELEASE)
	print_element (name.release);
      if (toprint & PRINT_KERNEL_VERSION)
	print_element (name.version);
      if (toprint & PRINT_MACHINE)
	print_element (name.machine);
    }

  /* 参数值==32 */
  if (toprint & PRINT_PROCESSOR)
    {
      char const *element = unknown;
#if HAVE_SYSINFO && defined SI_ARCHITECTURE
      {
	static char processor[257];
	if (0 <= sysinfo (SI_ARCHITECTURE, processor, sizeof processor))
	  element = processor;
      }
#endif
#ifdef UNAME_PROCESSOR
      if (element == unknown)
	{
	  static char processor[257];
	  size_t s = sizeof processor;
	  static int mib[] = { CTL_HW, UNAME_PROCESSOR };
	  if (sysctl (mib, 2, processor, &s, 0, 0) >= 0)
	    element = processor;
	}
#endif
      print_element (element);
    }

  /* 参数值==64 */
  if (toprint & PRINT_HARDWARE_PLATFORM)
    {
      char const *element = unknown;
#if HAVE_SYSINFO && defined SI_PLATFORM
      {
	static char hardware_platform[257];
	if (0 <= sysinfo (SI_PLATFORM,
			  hardware_platform, sizeof hardware_platform))
	  element = hardware_platform;
      }
#endif
#ifdef UNAME_HARDWARE_PLATFORM
      if (element == unknown)
	{
	  static char hardware_platform[257];
	  size_t s = sizeof hardware_platform;
	  static int mib[] = { CTL_HW, UNAME_HARDWARE_PLATFORM };
	  if (sysctl (mib, 2, hardware_platform, &s, 0, 0) >= 0)
	    element = hardware_platform;
	}
#endif
      print_element (element);
    }

  /* 参数值==128 */
  if (toprint & PRINT_OPERATING_SYSTEM)
    print_element (HOST_OPERATING_SYSTEM);

  putchar ('\n');

  exit (EXIT_SUCCESS);
}

用getopt_long解析用户传入参数后并保存在toprint,调用uname填充ustname结构体,判断torint输出对应的信息

问1:-a = -1时是怎么输出所有。是不是-1在&别的常量时if判断为true所以能输出所有。

因为发现5.0和新的源码差距太大了,所以man shuf翻到最下面查看了我linux的coreutils是8.32,所以去看8.32了,这个问题等8.32的时候在解决了

相关文章:

  • OpenGL入门(四)之纹理Texture
  • UDP和TCP协议发送接收数据
  • Apache Doris 快速学习大纲
  • FastFlow(5)---软件加速器 software accelerator
  • 华为OD:0019-0020:-最小步骤数—删除字符串中出现次数最少的字符
  • Python学生成绩管信息理系统(面向对象)(学生信息篇)
  • 国稻种芯百团计划行动 丰收节贸促会·袁隆平:水稻国际竞争
  • 面试精选:3、史上最详细的Linux精选面试题(二)
  • 2.21 haas506 2.0开发教程 - TTS - Text To Speech (320开发板)
  • Promethues-如何监控容器
  • 测试人生 | 从小团队的业务到独角兽的测开,涨薪超过60%,90后小哥哥凤凰涅槃了
  • 技术门槛高?来看 Intel 机密计算技术在龙蜥社区的实践
  • 532. 数组中的 k-diff 数对
  • 通过mybatis自定义参数类型转换器,进行数据库字段加密脱敏
  • Win10修复IPv6优先访问
  • android高仿小视频、应用锁、3种存储库、QQ小红点动画、仿支付宝图表等源码...
  • Asm.js的简单介绍
  • Fabric架构演变之路
  • leetcode讲解--894. All Possible Full Binary Trees
  • Linux各目录及每个目录的详细介绍
  • node 版本过低
  • python大佬养成计划----difflib模块
  • SpiderData 2019年2月16日 DApp数据排行榜
  • Vue组件定义
  • 不上全站https的网站你们就等着被恶心死吧
  • 从零开始的webpack生活-0x009:FilesLoader装载文件
  • 发布国内首个无服务器容器服务,运维效率从未如此高效
  • 分布式事物理论与实践
  • 官方解决所有 npm 全局安装权限问题
  • 技术攻略】php设计模式(一):简介及创建型模式
  • 紧急通知:《观止-微软》请在经管柜购买!
  • 前嗅ForeSpider采集配置界面介绍
  • 浅谈web中前端模板引擎的使用
  • 如何在GitHub上创建个人博客
  • 源码之下无秘密 ── 做最好的 Netty 源码分析教程
  • postgresql行列转换函数
  • 阿里云API、SDK和CLI应用实践方案
  • ​人工智能书单(数学基础篇)
  • ​云纳万物 · 数皆有言|2021 七牛云战略发布会启幕,邀您赴约
  • #免费 苹果M系芯片Macbook电脑MacOS使用Bash脚本写入(读写)NTFS硬盘教程
  • $L^p$ 调和函数恒为零
  • (11)工业界推荐系统-小红书推荐场景及内部实践【粗排三塔模型】
  • (2/2) 为了理解 UWP 的启动流程,我从零开始创建了一个 UWP 程序
  • (3)选择元素——(14)接触DOM元素(Accessing DOM elements)
  • (4) PIVOT 和 UPIVOT 的使用
  • (6)STL算法之转换
  • (Redis使用系列) Springboot 实现Redis消息的订阅与分布 四
  • (超详细)2-YOLOV5改进-添加SimAM注意力机制
  • (每日持续更新)信息系统项目管理(第四版)(高级项目管理)考试重点整理第3章 信息系统治理(一)
  • (十)c52学习之旅-定时器实验
  • (十八)SpringBoot之发送QQ邮件
  • (学习日记)2024.04.04:UCOSIII第三十二节:计数信号量实验
  • (原创)Stanford Machine Learning (by Andrew NG) --- (week 9) Anomaly DetectionRecommender Systems...
  • (转载)在C#用WM_COPYDATA消息来实现两个进程之间传递数据
  • .NET Compact Framework 多线程环境下的UI异步刷新