Skip to content

Commit

Permalink
Merge pull request #33 from schneems/fcheung-osx-ffi
Browse files Browse the repository at this point in the history
Test Faster version for Mac OS
  • Loading branch information
schneems authored Jul 8, 2019
2 parents c2aa830 + c84afc6 commit 890205c
Show file tree
Hide file tree
Showing 6 changed files with 85 additions and 4 deletions.
5 changes: 4 additions & 1 deletion .travis.yml
Original file line number Diff line number Diff line change
@@ -1,4 +1,7 @@
language: ruby
os:
- linux
- osx

before_install:
- gem install bundler -v 1.12.5
Expand All @@ -14,4 +17,4 @@ matrix:
allow_failures:
- rvm: ruby-head
- rvm: rbx-19mode
- rvm: jruby-19mode
- rvm: jruby-19mode
8 changes: 6 additions & 2 deletions CHANGELOG.md
Original file line number Diff line number Diff line change
@@ -1,6 +1,10 @@
## 0.2.4

- Use FFI to speed up memory lookup on Mac (darwin) by roughly 12x (#32)

## 0.2.3

- Silence BigDecimal deprecation warning under Ruby 2.5 (#26)
- Silence BigDecimal deprecation warning under Ruby 2.5 (#26)

## 0.2.2

Expand All @@ -25,4 +29,4 @@

## 0.0.1

- Initial
- Initial
2 changes: 2 additions & 0 deletions get_process_mem.gemspec
Original file line number Diff line number Diff line change
Expand Up @@ -18,6 +18,8 @@ Gem::Specification.new do |gem|
gem.test_files = gem.files.grep(%r{^(test|spec|features)/})
gem.require_paths = ["lib"]

gem.add_dependency "ffi", "~> 1.0"

gem.add_development_dependency "sys-proctable", "~> 1.0"
gem.add_development_dependency "rake", "~> 10.1"
gem.add_development_dependency "test-unit", "~> 3.1.0"
Expand Down
19 changes: 19 additions & 0 deletions lib/get_process_mem.rb
Original file line number Diff line number Diff line change
Expand Up @@ -31,6 +31,20 @@ class GetProcessMem
include Sys
end

RUNS_ON_DARWIN = Gem.platforms.detect do |p|
p.is_a?(Gem::Platform) && p.os == 'darwin'
end

if RUNS_ON_DARWIN
begin
require 'get_process_mem/darwin'
rescue LoadError => e
message = "Please add `ffi` to your Gemfile for darwin (macos) machines\n"
message << e.message
raise e, message
end
end

def initialize(pid = Process.pid)
@status_file = Pathname.new "/proc/#{pid}/status"
@process_file = Pathname.new "/proc/#{pid}/smaps"
Expand All @@ -44,6 +58,7 @@ def linux?

def bytes
memory = linux_status_memory if linux?
memory ||= darwin_memory if RUNS_ON_DARWIN
memory ||= ps_memory
end

Expand Down Expand Up @@ -102,4 +117,8 @@ def ps_memory
KB_TO_BYTE * number_to_bigdecimal(mem == "" ? 0 : mem)
end
end

def darwin_memory
Darwin.resident_size
end
end
53 changes: 53 additions & 0 deletions lib/get_process_mem/darwin.rb
Original file line number Diff line number Diff line change
@@ -0,0 +1,53 @@
require 'ffi'

class GetProcessMem
class Darwin
extend FFI::Library
ffi_lib 'c'
attach_function :mach_task_self, [], :__darwin_mach_port_t
attach_function :task_info,
[
:__darwin_mach_port_t,
:int, # return selector
:pointer, #pointer to task info
:pointer, #pointer to int (size of structure / bytes filled out)
],
:int

class IntPtr < FFI::Struct
layout :value, :int
end

class TaskInfo < FFI::Struct
layout :suspend_count, :int32,
:virtual_size, :uint64,
:resident_size, :uint64,
:user_time, :uint64,
:system_time, :uint64,
:policy, :int32
end

MACH_TASK_BASIC_INFO = 20
MACH_TASK_BASIC_INFO_COUNT = TaskInfo.size / FFI.type_size(:uint)

class << self
def resident_size
mach_task_info[:resident_size]
end

private

def mach_task_info
data = TaskInfo.new
out_count = IntPtr.new
out_count[:value] = MACH_TASK_BASIC_INFO_COUNT
result = task_info(mach_task_self, MACH_TASK_BASIC_INFO, data, out_count)
if result == 0
data
else
raise "task_info returned #{result}"
end
end
end
end
end
2 changes: 1 addition & 1 deletion lib/get_process_mem/version.rb
Original file line number Diff line number Diff line change
@@ -1,3 +1,3 @@
class GetProcessMem
VERSION = "0.2.3"
VERSION = "0.2.4"
end

0 comments on commit 890205c

Please sign in to comment.