This module is a native addon providing you with the ability to create a shared memory file and use it as Napi::Buffer
inside Node.JS!
It also provides you with a synchronization mechanism using flock to cordinate reads/writes between multiple processes using the same memory region.
For performance tuning, you are also provided with an option to memory lock the created region using mlock;
See also: shm_overview(7), mmap(2), flock(2), mlock(2)
- You should be aware of race condition that can happen when upgrading shared locks, to exclusive locks; If you are not, please look it up!
- Currently mlock uses
MLOCK_ONFAULT
flag for locking memory pages to avoid filling up your memory. - The
LOCK_NB
flag is not used and all locks are blocking at C level.
npm i mmap-ipc
Process #1:
const mmapIPC = require('mmap-ipc')
let mmap_id = "SharedBuf"; //This name is used to open the same buffer in different processes
let mmap_size = 4096; // aka. 4KB
let lock_page = true; // Wether to lock the memory page using mlock(2) MLOCK_ONFAULT
const SharedBuf = new mmapIPC(mmap_id, mmap_size, lock_page);
console.log( "Acquire write lock:" );
console.log( SharedBuf.acquireWriteLock() );
console.log( SharedBuf.buffer().write('Message from process #1', 'ascii') );
console.log( "Finished writing!" );
console.log( "Removing lock..." );
console.log( SharedBuf.removeLock() );
console.log( "Acquire read lock:" );
console.log( SharedBuf.acquireReadLock() );
console.log( SharedBuf.buffer().toString('ascii') );
console.log( "Finished reading!" );
console.log( SharedBuf.removeLock() );
console.log( "Removing lock..." );
Process #2:
const mmapIPC = require('mmap-ipc')
let mmap_id = "SharedBuf";
let mmap_size = 4096;
let lock_page = true;
const SharedBuf = new mmapIPC(mmap_id, mmap_size, lock_page);
console.log( "Acquire read lock:" );
console.log( SharedBuf.acquireReadLock() );
console.log( SharedBuf.buffer().toString('ascii') );
console.log( "Finished reading!" );
console.log( "Acquire write lock:" );
console.log( SharedBuf.acquireWriteLock() );
console.log( SharedBuf.buffer().write('This is process #2, I got your message!', 'ascii') );
console.log( "Finished writing!" );
console.log( "Removing lock..." );
console.log( SharedBuf.removeLock() );
To test this example, you can use a debugger and run the program step by step in 2 different processes as instructed by comments in the codes above!
[Constructor] mmapIPC(shm_name, size, lock_memory)
=> Buffer
On succuss return a new Node.JS Buffer
Using the specified shared memory as backing store!
Please note: Shared memory will be created and truncated to the specified size (see size
param below) if doesn't exist; If the specified shared memory already exists, it'll be open and won't be truncated!
It's your responsibily to provide the constructor with the correct size
(or a size
less then the actual size
of already existing shared memory) as it will not be truncated if already exist!
String shm_name
: A unique name used to identify and map a shared memory region between different processesNumeric size
: Size for the shared memory to be mapped; This is also the size of the returnedBuffer
. If a shared memory does not exist with the given name, a new one will be created, otherwise the already existing one will be mapped, but only as much bytes as specified with this option!Bool lock_memory
: Wether to lock the created/opened memory region with mlock(2). (see the note #2 above)
[Method] acquireReadLock()
=> Bool
Returns true
if acquiring the shared lock was successful, and will be blocked if needed until lock can be acquired!
This method accepts no parameter.
[Method] acquireWrireLock()
=> Bool
Returns true
if acquiring the exclusive lock was successful, and will be blocked if needed until lock can be acquired!
If a shared lock is being held by a previous call to acquireWrireLock()
, this call will upgrade that lock. (read more at the "see also" section above)
This method accepts no parameter.
[Method] removeLock()
=> Bool
Returns true
when the job's done!
A call to this method will release any lock that is currently being held.
This method accepts no parameter.
In case of any bug/question/suggestion please fill up an issue on Github.
Humbly yours, Th3R0b0t.