max_semi_space_size 设置值与实际值不一致的原因分析

简介:

问题由来

因为业务的需求,某 Node.js 性能平台用户需要调节新生代大小,Node.js 的启动参数里面的max_semi_space_size可以设置新生代堆空间的大小。

node --v8-options | grep max_semi -A 3 -B 2
  --min_semi_space_size (min size of a semi-space (in MBytes), the new space consists of twosemi-spaces)
        type: int  default: 0
  --max_semi_space_size (max size of a semi-space (in MBytes), the new space consists of twosemi-spaces)
        type: int  default: 0
  --semi_space_growth_factor (factor by which to grow the new space)
        type: int  default: 2

相关文档里该值是一个以MB为单位的整数,并没有其他特别的约束。

然而,当用户设置 max_semi_space_size 为200时,Node.js 性能平台 GC Trace分析 结果显示 inactive new_space semispace:256MB,说明新生代中未使用的那部分是256MB(有关GC的一些知识可以参阅文档最后的列表),如下图所示。

undefined

我们也线下验证了一下:

  • 当设置 max_smi_space_size 为100,110时,inactive new_space semispace:128MB
  • 当设置 max_smi_space_size 为50,60时,inactive new_space semispace:64MB

那么问题来了,是 Node.js 性能平台的bug?还是v8引擎本身的设计就是如此?

问题定位

其实,从最终64/128/256,这些数值就能推测到,设计就是如此。在计算机的世界里,2的整数次幂会带来很多方便。

还是老方法,既然文档里没有明确说明,而 Node.js 性能平台运行时在该部分功能跟社区运行时是完全一致的,那么只能去代码里找原因了。
一番操作之后,在 deps/v8/src/heap/heap.cc 中找到函数:

bool Heap::ConfigureHeap(size_t max_semi_space_size, size_t max_old_space_size, size_t code_range_size))。

这里摘抄一点max_semi_space相关代码,相关注释直接写到代码里面了,感兴趣的同学建议去看看完整代码。

bool Heap::ConfigureHeap(size_t max_semi_space_size, size_t max_old_space_size,
                         size_t code_range_size) {
  if (HasBeenSetUp()) return false;

  // Overwrite default configuration.
  // 未设置 max_semi_space_size 时,默认值是 0 
  if (max_semi_space_size != 0) {
    max_semi_space_size_ = max_semi_space_size * MB;
  }
  ...

  // If max space size flags are specified overwrite the configuration.
  // 命令行 --max_semi_space_size 设置的新生代大小是通过 FLAG_max_semi_space_size 传递到v
  if (FLAG_max_semi_space_size > 0) {
    max_semi_space_size_ = static_cast<size_t>(FLAG_max_semi_space_size) * MB;
  }
  ...

  /* 
  ROUND_UP 的定义:
  // Round up n to be a multiple of sz, where sz is a power of 2.
     #define ROUND_UP(n, sz) (((n) + ((sz) - 1)) & ~((sz) - 1))
  */
  // 操作系统相关的内存页大小,我的ubuntu16.04上该值是 512KB
  if (Page::kPageSize > MB) {
    max_semi_space_size_ = ROUND_UP(max_semi_space_size_, Page::kPageSize);
    ...
  }
  // 该参数默认是false
  if (FLAG_stress_compaction) {
    // This will cause more frequent GCs when stressing.
    max_semi_space_size_ = MB;
  }

  // The new space size must be a power of two to support single-bit testing
  // for containment.
  // 关键点在这里
  // 为什么 50 变成了 64, 100/120 变成了128, 200 变成了 256
  // 下面的函数 RoundUpToPowerOfTwo32 就是这个变化的原因。
  /*
uint32_t RoundUpToPowerOfTwo32(uint32_t value) {
  DCHECK_LE(value, uint32_t{1} << 31);
  if (value) --value;
// Use computation based on leading zeros if we have compiler support for that.
#if V8_HAS_BUILTIN_CLZ || V8_CC_MSVC
  return 1u << (32 - CountLeadingZeros32(value));
#else
  value |= value >> 1;
  value |= value >> 2;
  value |= value >> 4;
  value |= value >> 8;
  value |= value >> 16;
  return value + 1;
#endif
}
  */ 
  max_semi_space_size_ = base::bits::RoundUpToPowerOfTwo32(
      static_cast<uint32_t>(max_semi_space_size_));
  // 这里是 min_semi_space_size 的设置,这里不讨论
  if (FLAG_min_semi_space_size > 0) {
    ...
  }
  
  // 新生代初始大小是 min_semispace_size,如果需要,那么增大到 max_semi_space_size
  initial_semispace_size_ = Min(initial_semispace_size_, max_semi_space_size_);

  if (FLAG_semi_space_growth_factor < 2) {
    FLAG_semi_space_growth_factor = 2;
  }

  ...

  configured_ = true;
  return true;
}

问题结论

max_semi_space_size 设置看起来是个任意整数,但是实际使用中 v8 会把该值转换成一个不小于该值的2的整数次幂的值。也就是说:

  • max_semi_space_size 设置为 33, 34, ..., 64,最终结果都是 64MB。
  • max_semi_space_size 设置为 65, 66, ..., 128,最终结果都是 128MB。
  • 依次类推

heap.cc 里面的注释是

  // The new space size must be a power of two to support single-bit testing
  // for containment.

相关知识

目录
相关文章
|
11月前
|
人工智能 自然语言处理 Java
DeepSeek 满血版在 IDEA 中怎么用?手把手教程来了
DeepSeek 满血版在 IDEA 中怎么用?手把手教程来了
|
监控 安全 中间件
Next.js 实战 (十):中间件的魅力,打造更快更安全的应用
这篇文章介绍了什么是Next.js中的中间件以及其应用场景。中间件可以用于处理每个传入请求,比如实现日志记录、身份验证、重定向、CORS配置等功能。文章还提供了一个身份验证中间件的示例代码,以及如何使用限流中间件来限制同一IP地址的请求次数。中间件相当于一个构建模块,能够简化HTTP请求的预处理和后处理,提高代码的可维护性,有助于创建快速、安全和用户友好的Web体验。
329 0
Next.js 实战 (十):中间件的魅力,打造更快更安全的应用
|
JavaScript
js如何添加新元素到数组中
js如何添加新元素到数组中
486 0
|
存储 缓存 资源调度
Vue3状态管理新选择:Pinia安装与使用详解,以及与Vuex的对比分析
Vue3状态管理新选择:Pinia安装与使用详解,以及与Vuex的对比分析
540 0
|
JavaScript Java 开发工具
Electron V8排查问题之接近堆内存限制的处理如何解决
Electron V8排查问题之接近堆内存限制的处理如何解决
882 1
|
Kubernetes 开发者 Perl
使用kubectl port-forward端口转发来快速调试应用
通过使用 `kubectl port-forward`,开发者能够直接从本地机器访问和调试在Kubernetes集群内运行的服务,这是快速反馈和故障排除的利器。
1026 0
|
定位技术
vue-baidu-map 自定义地图主题
vue-baidu-map 自定义地图主题
430 0
|
设计模式 自然语言处理 算法
摆脱复杂图谱术语,7个原则搞定Schema建模
本文我们结合蚂蚁域内的多个业务场景,举例说明结合SPG规范的结构与语义解耦的知识建模及schema设计方法。
|
JavaScript 数据安全/隐私保护
vue项目实战之点击小眼睛实现input密码框内容显示与隐藏
vue项目实战之点击小眼睛实现input密码框内容显示与隐藏
vue项目实战之点击小眼睛实现input密码框内容显示与隐藏
|
Java
IDEA代码生成插件CodeMaker
## 前言 Java 开发过程中经常会遇到编写重复代码的事情,例如说:编写领域类和持久类的时候,大部分时候它们的变量名称,类型是一样的,在编写领域类的时候常常要重复写类似的代码。类似的问题太多,却没找到可以支持自定义代码模板的插件,只能自己动手,丰衣足食,开发了一个 IDEA 的代码生成插件,通过 Velocity 支持自定义代码模板来生成代码。
5162 0