18
Sat, Jan
2 New Articles

The ABCs of MI Locks

RPG
Typography
  • Smaller Small Medium Big Bigger
  • Default Helvetica Segoe Georgia Times

It's time for a review of the basic concepts of MI locks. These examples will help.

 

In computer science, a lock is a synchronization mechanism for enforcing limits on access to a resource in an environment where there are many threads of execution. In IBM i, locks are implemented in such a complex and sophisticated manner that they cannot be compared directly with their counterparts in common platforms, such as semaphores, critical sections, read/write locks, and mutexes.

 

The design of IBM i locks allows OS applications and user programs to achieve fine-grained control of resources and escape some common problems with lock implementations. At the MI level, some of the complex and versatile locking support is arranged and exposed to user programs in the form of a bunch of lock management MI instructions. This article reviews the basic concepts of MI locks by looking at some easy-to-follow examples and experiments.

 

In this article, I refer to an IBM i job (or an MI process at the MI level) simply as process; the term lock requester means a process or a thread; and the term lock target means an MI object lock or a space location to lock.

 

More examples of using MI lock instructions written in ILE RPG can be found here.

Types of MI Locks

IBM allows user programs to acquire or release the following types of MI locks:

 

The most significant difference between these two types of MI locks is that object locks are at the granularity of an MI object, while space location locks are at the granularity of each single byte in the associated space of an MI object.

 

(Note: The Lock Teraspace Storage Location (LOCKTSL) and Unlock Teraspace Storage Location (UNLCKTSL) can be used to lock/unlock either a Teraspace location or a Single Level Store (SLS) space location. For reasons of simplicity, these instructions and Teraspace location locks are not covered in this article.)

 

Here's an explanation of the common lock allocation procedure of MI locks:

  • A single lock-acquisition instruction (LOCK or LOCKSL) can request the allocation of one or more lock states on one or more lock targets (MI objects or space locations). Locks are allocated sequentially until all locks requested are allocated.
  • When two or more threads are competing for a conflicting lock allocation on a lock target, the machine attempts to first satisfy the lock allocation request of the thread with the highest priority. Within that priority, the machine attempts to satisfy the request that has been waiting longest.
  • If any exception is identified during the instruction's execution, any locks already granted by the instruction are released, and the lock request is canceled.
  • For each object lock, counts are kept by lock state and by thread, process, or transaction control structure. When a lock request is granted, the appropriate lock count of each lock state specified is incremented by 1. For each space location lock, counts are kept by lock state and by thread. If a transfer of an object lock from another thread causes a previously unsatisfied lock request to become satisfied, the lock request and the transfer lock are treated independently relative to lock accounting. The appropriate lock counts are incremented for both the lock request and the transfer lock function.

Lock States

Unlike common lock mechanisms, such as mutexes or read/write locks in which lock states are very limited, MI locks can request one or more of all the following five lock states on the lock targets:

  • LSRD—The lock target can be shared with another lock requester if the user does not request exclusive use of the lock target. That is, another user can request a LSRO, LSUP, LEAR, or LSRD lock state.
  • LSRO—The lock target can be shared with another lock requester if the lock requester requests either a LSRO lock state or a LSRD lock state. This lock state is appropriate when a user does not intend to change a lock target but wants to ensure that no other user changes the lock target.
  • LSUP—The lock target can be shared either for update or read with another lock requester. That is, another user can request either a LSRD lock state or a LSUP lock state for the same lock target. This lock state is appropriate when a user intends to change a lock target but wants to allow other users to read or change the same lock target.
  • LEAR—The lock target is allocated to the lock requester that requests it, but other lock requesters can read the lock target. This lock is appropriate when a user wants to prevent other users from performing any operation other than a read.
  • LENR—The lock target is reserved for the exclusive use of the requesting lock requester; no other lock requesters can use the lock target. However, if the lock target is already allocated to another lock requester, you cannot get exclusive use of the lock target. This lock state is appropriate when a user does not want any other user to have access to the lock target until the function being performed is complete.

 

The following table shows the valid lock state combinations for an object.

 

Valid Lock State Combinations

Lock state held by one lock requester

Another lock requester can obtain this lock state

LENR

None

LEAR

LSRD

LSUP

LSUP or LSRD

LSRO

LSRO or LSRD

LSRD

LEAR, LSUP, LSRO, or LSRD

 

The design of the multiple lock states is important. First, this design can help lower the performance overhead due to lock contention, which is a common problem with usual lock mechanisms. For example, if a lock requester requests only LSRD on a lock target, no lock contention will occur between it and other lock requesters who request a LSRO, LSUP, or LEAR lock on the same lock target. Second, this design can help user programs and OS applications avoid deadlocks, which are more common in other platforms, where most lock mechanisms available are mutually exclusive locks.

 

It's interesting that, except for the Lock Management MI instructions, the only interfaces IBM provides to user programs to acquire or release an MI lock are CL commands: the Allocate Object (ALCOBJ) command and the Deallocate Object (DLCOBJ) command. Users can use them to acquire or release object locks conveniently either interactively or in programs. These commands use a set of special values to indicate different lock states. For convenience, I list the mapping of MI lock state names and lock state special values used by ALCOBJ and DLCOBJ in the table below.

 

 

Mapping from MI Lock State Names to CL Lock State Special Values

MI Lock State

CL Lock State

LSRD

*SHRRD

LSRO

*SHRNUP

LSUP

*SHRUPD

LEAR

*EXCLRD

LENR

*EXCL

 

To obtain the mapping table shown above, you can follow a simple experiment like this:

 

1. Select an MI object as your lock target (say, a program object called *LIBL/MAPMAP).

2. Obtain the 8-byte SLS virtual address of MAPMAP by dumping it (via the Dump Object (DMPOBJ) command or the Dump System Object (DMPSYSOBJ)) or find it in the SST via object name and MI object type/subtype code. Let's say the address of MAPMAP is hex 060A6CEBB4000000.

3. Acquire a specific lock state on MAPMAP using the ALCOBJ command. For example, acquire a LENR lock by ALCOBJ ((MAPMAP *PGM *EXCL)).

4. Examine the lock record via the advanced analysis macro LOCKINFO of the System Service Tool (SST)—for example, LOCKINFO 060A6CEBB4000000. Check the MI lock state name in the output of macro LOCKINFO, and you will discover that *EXCL means MI lock state LENR. An example output is the following:

 

 

     DISPLAY/ALTER/DUMP                                               12/04/09  17:44:16   PAGE      1

Running macro: LOCKINFO                         060A6CEBB4000000                                                                  

                                                                                                                                   

Dumping lock table                                                                            

Hold Records in HoldHashTableEntryBlock C00000200000FB00 for object address 060A6CEBB4000000    

 Task B000C00002AC8000 holds 1 Process scoped LENR Lock NOT Monitored

187C871D81003FBC-progrm QSYS/QDMLOCK+0x2A3C Inst# 0x1E6     

                                                                                                                                   

lockinfo complete

 

5. Release the acquired lock on MAPMAP via the DLCOBJ command. For example, DLCOBJ ((MAPMAP *PGM *EXCL)).

6. Repeat steps 1 through step 5 for each of other lock states.

Lock Request Types

With a single LOCK or LOCKSL instruction, a lock requester can request one or more lock states on one or more MI objects or space locations. An MI lock request can be one of the following types:

  • Immediate request—If all locks cannot be immediately granted, a Lock Request Not Grantable (hex 1A02) exception (aka MCH2602) will be raised.
  • Synchronous request—Wait until all locks can be granted for a specified time interval or infinitely.
  • Asynchronous request (available only for MI object locks)—Allow processing to continue and signal event when the object is available.

 

For an asynchronous object lock request, if the asynchronous lock request is satisfied, then the Object Locked (hex 000A,01,01) event is signaled to the requesting thread. If the request is not satisfied in the specified time interval, the Asynchronous Lock Wait Timeout (hex 000A,04,01) event is signaled to the requesting thread. No locks are granted, and the lock request is canceled. If an object is destroyed while a thread has a pending request to lock the object, the Object Destroyed (hex 000A,02,01) event is signaled to the waiting thread. These events are signaled to the requesting thread, regardless of the scope of the requested lock. If the invocation issuing the asynchronous object lock request is terminated, the lock request remains active.

 

Let's experiment with different lock request tests using the following example program, lock01.rpgle.

 

     h dftactgrp(*no)

      /copy mih-lock

      /copy mih-ptr

     d main            pr                  extpgm('LOCK01')

     d   request_type                 1a

 

     d lock_request    ds                  qualified

     d   base                              likeds(lock_request_tmpl_t)

     d   obj                           *

     d   lock_state                   1a

 

     d main            pi

     d   request_type                 1a

 

      /free

           lock_request.base = *allx'00';

           lock_request.base.num_requests = 1;

           lock_request.base.offset_lock_state = 32;

           lock_request.base.lock_opt = x'4200'; // Synchronous request

                                                 // and wait indefinitely.

           // Lock myself

           rslvsp_tmpl.obj_type = x'0201';

           rslvsp_tmpl.obj_name = 'LOCK01';

           rslvsp2(lock_request.obj : rslvsp_tmpl);

           lock_request.lock_state = x'09';      // LENR lock

           if request_type = 'I';

               lock_request.base.lock_opt = x'0200'; // Immediate request

           elseif request_type = 'A';

               lock_request.base.lock_opt = x'8200'; // Asynchronous request

           endif;

 

           lockobj(lock_request);  // Lock myself with specified lock request type

           *inlr = *on;

      /end-free

 

For example, you may start two interactive jobs: JOB#A and JOB#B. Type the following commands in JOB#A and JOB#B.

 

JOB#A

JOB#B

CALL LOCK01 X

 

 

CALL LOCK01 X  /* Request a LENR lock on *PGM LOCK01 synchronously */

 

CALL LOCK01 I  /* Request an immediate LENR lock on *PGM LOCK01 */

 

CALL LOCK01 A  /* Request a LENR lock on *PGM LOCK01 asynchronously */

 

The Work with Object Locks (WRKOBJLCK) command is helpful when checking current lock status of a given object (including acquired lock states or lock states being requested synchronously or asynchronously). For example, when JOB#B requests an asynchronous LENR lock on *PGM LOCK01, the output of command WRKOBJLCK LOCK01 *PGM might like this:

 

Opt   Job          User         Lock      Status          Scope     Thread  

 _    JOB#A        LJL          *EXCL      HELD           *JOB              

 _    JOB#B        LJL          *EXCL      REQ            *JOB      00000064

Lock Scopes

In common platforms, locks are usually scoped to thread, while MI locks may be scoped to three different lock scopes. Space location locks acquired by LOCKSL can only be scoped to thread; object locks and space location locks acquired by LOCKTSL can be scoped to a thread, a process, or a transaction control structure. Locks scoped to a thread can never conflict with a lock scoped to its containing process, but may conflict with a lock scoped to a different process, a transaction control structure, or any other thread (depending on the lock states involved). Locks scoped to a transaction control structure attached to the current thread may conflict with a lock scoped to a different transaction control structure, a process, or any other thread (depending on the lock states involved).

 

What's a Transaction Control Structure (TCS) object? It's an MI object type whose MI object type code is hex 23. You can find out the sub-object types of TCS via the lsobjtypes QShell utility provided by the i5/OS Programmer's Toolkit.

 

> lsobjtypes | grep '^23'

  23A1  CDJOBLK          # Job/activation-group level commit definition or XA

                               commitment definition with job scoped locks

  23A0  CDTCSLK          # XA commitment definition with transaction scoped locks

  2300  TCS               # Transaction Control Structure

  $

 

Note: Common job-level or activation-group level commitment definitions are also represented by the 23A1 TCS object since the introduction of the TCS MI object.

 

For detailed information about XA support provided by IBM i, please refer to documentation on XA APIs in the IBM i Information Center.

 

A nice article written by Jarek Miszczyk in 2007 is also available at MC Press Online: "Bridge the Legacy-to-Java Transition with DB2 for i5/OS Distributed Transactions."

 

Allocated process scope locks are released when the process terminates. Allocated thread scope locks are released when the thread terminates. If a thread requested a process scope lock, the process will continue to hold that lock after termination of the requesting thread. If a thread requested a transaction control structure scope lock, the transaction control structure will continue to hold that lock after the termination of the requesting thread. If a thread is terminated while waiting for a lock with a lock request type of either synchronous or asynchronous, the lock request is canceled regardless of the scope of the requested lock.

Authority Required to Acquire MI Object Locks

To acquire a space location lock via a valid space pointer, the lock requester need not have any authority to the MI object that the space pointer addresses. But to acquire an object lock, the condition is somewhat different:

  • First, the lock requester needs Execute authority to the contexts (libraries) referenced for address resolution.
  • Second, the lock requester needs some authority to the target object.

 

The document of the ALCOBJ command says: The user issuing the command must have object operational (*OBJOPR) authority to the object.

 

The document of the LOCK instruction in the information center says: Authorization required to the object to be locked: Some authority or ownership.

 

When an MI object is created via CL commands or APIs, full authorities to the object are always granted to its owner user profile. Then, what does some authority mean? If you are really curious about it, try a simple experiment like the following:

  • Say program object I#PGM is owned by user USER#A, and its public authority is set to *EXCLUDE.
  • Say another user, USER#B, has no source of authority to obtain any authority to program I#PGM (e.g., via authorization lists or primary group (PGP)).
  • Grant any one of the object authorities or data authorities to USER#B on program I#PGM by the Grant Object Authority (GRTOBJAUT) command or the Edit Object Authority (EDTOBJAUT) command. Object authorities to an object are *OBJEXIST, *OBJMGT, *OBJOPR, *OBJALTER, *OBJREF, and *AUTL; data authorities to an object are *READ, *ADD, *DLT, *UPD, and *EXECUTE. For example, GRTOBJAUT OBJ(*LIBL/I#PGM) OBJTYPE(*PGM) USER(USER#B) AUT(*READ).
  • Finally, try to lock I#PGM in an interactive job under user profile USER#B. For example, ALCOBJ ((*LIBL/I#PGM *PGM *EXCL)).

 

Now, you should have the answer!

Advisory Locks or Mandatory Locks?

In common platforms, most locks are advisory locks, where each thread cooperates by acquiring the lock before accessing the corresponding data. However, some platforms also implement mandatory locks, where attempts of unauthorized access to a locked resource will force an exception in the entity attempting to make the access. So, are MI locks advisory locks or mandatory locks?

 

Space location locks are advisory locks. No matter what states of lock are being allocated by a thread on a space location, another thread can do whatever it wants to the space location: changing data content at the space location, extending or truncating the space object, and even destroying the space object.

 

Unlike space location locks, in a sense, MI object locks are mandatory locks. Locks are not required in order for an object to be referred to (or modified by) an instruction. However, a process or thread is not allowed to use an object if any other process or other thread in the same process holds a conflicting lock. This is referred to as Lock Enforcement Rules in MI documentation. For example, when a LSRO lock on the owner user profile of a permanent space object or the context the space object resides in is held by another thread, a thread issuing a Destroy Space (DESS) MI instruction on the space object will cause an Invalid Lock State (hex 1A01) exception (aka MCH2601), which means one or more lock enforcement rules are violated when an attempt is made to access an MI object. Another example occurs in the previous section of this article, Lock Request Types, where although JOB#A holds a LENR lock on program object LOCK01, JOB#B can still call it as usual; this is because no lock enforcement rule is enforced on the Call External (CALLX) instruction of the called program object.

 

For lock enforcement rules enforced on an individual MI instruction, please refer to the Lock Enforcement section in the MI instruction's documentation.

 

Note that it's a good convention to acquire proper lock states on a shared object before actually accessing it so that the lock enforcement rules can help you to guarantee the integrity of your data and application systems. APIs and CL commands that access objects are a good example of this. For example, when a HLL program opens a physical file for reading, the QSYS/QDMCOPEN API will acquire a LSRO lock on the *FILE object (with MI object type code/sub-type code hex 1901), and the QSYS/QDBOPEN API will acquire a LSRO lock on the data space object (*QDDS, with MI object type code/subtype code hex 0B90) of the physical file.

Object Lock Transferring

As mentioned above, when multiple threads are competing for a conflicting object lock, the system satisfies one of the multiple lock requests according to thread priorities and waiting time of the threads. Besides the system coordinating mechanism, IBM i provides an alternative method to allow a process to transfer an object lock it holds to another process. After the transferring, the latter process becomes the new holder of the object lock. The MI instruction we can use to achieve object lock transferring is Transfer Object Lock (XFRLOCK). The first operand of XFRLOCK is a system pointer to the receiving process's Process Control Space (PCS) object.

 

Here, I'd like to demonstrate the use of object lock transferring with a simple and interesting experiment. Imagine we have a User Space (*USRSPC) object called BALL, and we want BALL available to only one of a couple of jobs (JOB#A and JOB#B) at any given time. We'll let one of the two jobs, say JOB#A, hold a LENR lock on BALL and then transfer the LENR lock on BALL to JOB#B when JOB#B needs to use BALL. Thus, the LENR lock on BALL is transferred between JOB#A and JOB#B repeatedly, and no other job can have the opportunity to acquire any kind of lock on BALL. Objects involved in this experiment are the following:

  • *USRSPC BALL
  • *USRSPC PCSPTR—The PCS pointer of the job to receive the LENR lock on BALL is stored at the beginning of PCSPTR's associated space.
  • *PGM LOCK03—LOCK03 writes the PCS pointer of the current process to the beginning of the associated space of PCSPTR.
  • *PGM LOCK02—LOCK02 transfers the LENR lock on BALL to the process identified by the PCS pointer stored in the associated space of PCSPTR.

 

Here's the source of OPM MI program LOCK03, lock03.emi:

 

/**

 * @file lock03.emi

 *

 * Saves the PCS pointer of the current process in 1934 space PCSPTR

 */

dcl sysptr space auto init (

        "PCSPTR", type(spc, h'34')

)                               ;

 

dcl dd tmpl char(32) auto bdry(16)  ;

dcl spcptr tmpl-ptr auto init(tmpl) ;

dcl spc * bas(tmpl-ptr)             ;

        dcl dd bin bin(4) dir       ;

        dcl dd bout bin(4) dir      ;

        dcl dd * char(8) dir        ;

        dcl sysptr pcs-ptr dir      ;

 

        cpynv bin, 32               ;

        matpratr tmpl-ptr, *, x'25' ;

brk '1'                             ;

 

dcl spcptr space-ptr auto       ;

dcl spc pcs-t bas(space-ptr)    ;

        dcl sysptr tgt-ptr dir  ;

 

        rslvsp space, *, *, *   ;

        setsppfp space-ptr, space ;

        cpybwp tgt-ptr, pcs-ptr   ;

brk 'END'                       ;

        rtx *                   ;

pend                            ;

 

 

Here's the source of ILE RPG program LOCK02, lock02.rpgle:

 

     h dftactgrp(*no)

      /copy mih-lock

      /copy mih-ptr

     d main            pr                  extpgm('LOCK02')

     d   obj_name                    30a

     d   obj_type                     2a

 

      * System pointer to *USRSPC PCSPTR

     d                 ds

     d spcobj@                         *   procptr

     d spcobj                          *   overlay(spcobj@)

      * Space pointer addresses the associated space of *USRSPC PCSPTR

     d spp             s               *

     d                 ds                  based(spp)

     d   pcs                           *

 

     d lock_request    ds                  qualified

     d   base                              likeds(lock_request_tmpl_t)

     d   obj                           *

     d   lock_state                   1a

 

     d main            pi

     d   obj_name                    30a

     d   obj_type                     2a

 

      /free

           // Lock transfer request template

           lock_request.base = *allx'00';

           lock_request.base.num_requests = 1;

           lock_request.base.offset_lock_state = 32;

           lock_request.base.lock_opt = x'0000';

           rslvsp_tmpl.obj_type = obj_type;

           rslvsp_tmpl.obj_name = obj_name;

           rslvsp2(lock_request.obj : rslvsp_tmpl);

           lock_request.lock_state = x'09';      // LENR lock

 

           // Retrieve the PCS pointer of the receiving process

           rslvsp_tmpl.obj_type = x'1934';

           rslvsp_tmpl.obj_name = 'PCSPTR';

           rslvsp2(spcobj : rslvsp_tmpl);

           spp = setsppfp(spcobj@);

 

           // Transfer the lock to the receiving process

           xfrlock(pcs : lock_request);

           *inlr = *on;

      /end-free

 

Now, let's start our ball-passing experiment. Use the Work with Object Lock (WRKOBJLCK) command to check which job is currently holding the lock on BALL.

 

Steps of the BALL-Passing Experiment

JOB#A

JOB#B

Remark

ALCOBJ ((BALL *USRSPC *EXCL))

 

JOB#A acquires an LENR lock on BALL

 

CALL LOCK03

JOB#B stores its PCS pointer in PCSPTR

CALL LOCK02 (BALL X'1934')

 

JOB#A transfers the lock to JOB#B

CALL LOCK03

 

JOB#A stores its PCS pointer in PCSPTR

 

CALL LOCK02 (BALL X'1934')

JOB#B transfers the lock to JOB#A

 

 

Now the lock is held by JOB#A again

Deadlocks

Deadlocks are a well-known problem with most lock mechanisms, especially where locks are always mutually exclusive locks. Software failures due to deadlocks are hard to detect or reproduce. The design of MI lock states is helpful in avoiding this problem, but the possibility of deadlocks still remains. The following is a simple example. Imagine we have two CL programs: PGM#A and PGM#B.

 

PGM#A

             ALCOBJ     OBJ((SPCA *USRSPC *SHRNUP))

             DLYJOB     DLY(10)

             ALCOBJ     OBJ((SPCB *USRSPC *SHRUPD)) WAIT(32767)

 

PGM#B

             ALCOBJ     OBJ((SPCB *USRSPC *SHRNUP))

             DLYJOB     DLY(10)

             ALCOBJ     OBJ((SPCA *USRSPC *SHRUPD)) WAIT(32767)

 

Submit two jobs like the following:

SBMJOB CMD(CALL PGM#A) JOB(JOB#A)

SBMJOB CMD(CALL PGM#B) JOB(JOB#B)

 

The result is that either of the two submitted jobs enters lock-waiting status. The object lock statuses of *USRSPC SPCA and SPCB are the following.

 

SPCA

Opt   Job          User         Lock      Status          Scope     Thread 

      JOB#A        LJL          *SHRNUP    HELD           *JOB             

      JOB#B        LJL          *SHRUPD    WAIT           *JOB      0000017E

 

SPCB

Opt   Job          User         Lock      Status          Scope     Thread 

      JOB#A        LJL          *SHRUPD    WAIT           *JOB      0000011D

      JOB#B        LJL          *SHRNUP    HELD           *JOB             

 

JOB#A is waiting for a LSUP lock on SPCB, which conflicts with the LSRO lock on SPCB held by JOB#B. And JOB#B is waiting for a LSUP lock on SPCA, which conflicts with the LSRO lock on SPCA held by JOB#A. This is a deadlock condition, since neither of the lock requests can be satisfied and neither of the two jobs will have the opportunity to recover from the deadlock.

 

So, as you might have already guessed, it's a good idea to avoid acquiring unnecessarily strict locks on lock targets (for example, a LENR lock when what you need is only to change the data content of your lock target). Stricter locks are more likely to cause blocking or deadlocks.

 

Additionally, to avoid deadlocks on a group of logically related lock targets, it will be helpful to standardize the lock acquisition sequences so that locks on the lock targets are acquired and released in the same order for each thread that uses this group of lock targets.

 

BLOG COMMENTS POWERED BY DISQUS

LATEST COMMENTS

Support MC Press Online

$

Book Reviews

Resource Center

  • SB Profound WC 5536 Have you been wondering about Node.js? Our free Node.js Webinar Series takes you from total beginner to creating a fully-functional IBM i Node.js business application. You can find Part 1 here. In Part 2 of our free Node.js Webinar Series, Brian May teaches you the different tooling options available for writing code, debugging, and using Git for version control. Brian will briefly discuss the different tools available, and demonstrate his preferred setup for Node development on IBM i or any platform. Attend this webinar to learn:

  • SB Profound WP 5539More than ever, there is a demand for IT to deliver innovation. Your IBM i has been an essential part of your business operations for years. However, your organization may struggle to maintain the current system and implement new projects. The thousands of customers we've worked with and surveyed state that expectations regarding the digital footprint and vision of the company are not aligned with the current IT environment.

  • SB HelpSystems ROBOT Generic IBM announced the E1080 servers using the latest Power10 processor in September 2021. The most powerful processor from IBM to date, Power10 is designed to handle the demands of doing business in today’s high-tech atmosphere, including running cloud applications, supporting big data, and managing AI workloads. But what does Power10 mean for your data center? In this recorded webinar, IBMers Dan Sundt and Dylan Boday join IBM Power Champion Tom Huntington for a discussion on why Power10 technology is the right strategic investment if you run IBM i, AIX, or Linux. In this action-packed hour, Tom will share trends from the IBM i and AIX user communities while Dan and Dylan dive into the tech specs for key hardware, including:

  • Magic MarkTRY the one package that solves all your document design and printing challenges on all your platforms. Produce bar code labels, electronic forms, ad hoc reports, and RFID tags – without programming! MarkMagic is the only document design and print solution that combines report writing, WYSIWYG label and forms design, and conditional printing in one integrated product. Make sure your data survives when catastrophe hits. Request your trial now!  Request Now.

  • SB HelpSystems ROBOT GenericForms of ransomware has been around for over 30 years, and with more and more organizations suffering attacks each year, it continues to endure. What has made ransomware such a durable threat and what is the best way to combat it? In order to prevent ransomware, organizations must first understand how it works.

  • SB HelpSystems ROBOT GenericIT security is a top priority for businesses around the world, but most IBM i pros don’t know where to begin—and most cybersecurity experts don’t know IBM i. In this session, Robin Tatam explores the business impact of lax IBM i security, the top vulnerabilities putting IBM i at risk, and the steps you can take to protect your organization. If you’re looking to avoid unexpected downtime or corrupted data, you don’t want to miss this session.

  • SB HelpSystems ROBOT GenericCan you trust all of your users all of the time? A typical end user receives 16 malicious emails each month, but only 17 percent of these phishing campaigns are reported to IT. Once an attack is underway, most organizations won’t discover the breach until six months later. A staggering amount of damage can occur in that time. Despite these risks, 93 percent of organizations are leaving their IBM i systems vulnerable to cybercrime. In this on-demand webinar, IBM i security experts Robin Tatam and Sandi Moore will reveal:

  • FORTRA Disaster protection is vital to every business. Yet, it often consists of patched together procedures that are prone to error. From automatic backups to data encryption to media management, Robot automates the routine (yet often complex) tasks of iSeries backup and recovery, saving you time and money and making the process safer and more reliable. Automate your backups with the Robot Backup and Recovery Solution. Key features include:

  • FORTRAManaging messages on your IBM i can be more than a full-time job if you have to do it manually. Messages need a response and resources must be monitored—often over multiple systems and across platforms. How can you be sure you won’t miss important system events? Automate your message center with the Robot Message Management Solution. Key features include:

  • FORTRAThe thought of printing, distributing, and storing iSeries reports manually may reduce you to tears. Paper and labor costs associated with report generation can spiral out of control. Mountains of paper threaten to swamp your files. Robot automates report bursting, distribution, bundling, and archiving, and offers secure, selective online report viewing. Manage your reports with the Robot Report Management Solution. Key features include:

  • FORTRAFor over 30 years, Robot has been a leader in systems management for IBM i. With batch job creation and scheduling at its core, the Robot Job Scheduling Solution reduces the opportunity for human error and helps you maintain service levels, automating even the biggest, most complex runbooks. Manage your job schedule with the Robot Job Scheduling Solution. Key features include:

  • LANSA Business users want new applications now. Market and regulatory pressures require faster application updates and delivery into production. Your IBM i developers may be approaching retirement, and you see no sure way to fill their positions with experienced developers. In addition, you may be caught between maintaining your existing applications and the uncertainty of moving to something new.

  • LANSAWhen it comes to creating your business applications, there are hundreds of coding platforms and programming languages to choose from. These options range from very complex traditional programming languages to Low-Code platforms where sometimes no traditional coding experience is needed. Download our whitepaper, The Power of Writing Code in a Low-Code Solution, and:

  • LANSASupply Chain is becoming increasingly complex and unpredictable. From raw materials for manufacturing to food supply chains, the journey from source to production to delivery to consumers is marred with inefficiencies, manual processes, shortages, recalls, counterfeits, and scandals. In this webinar, we discuss how:

  • The MC Resource Centers bring you the widest selection of white papers, trial software, and on-demand webcasts for you to choose from. >> Review the list of White Papers, Trial Software or On-Demand Webcast at the MC Press Resource Center. >> Add the items to yru Cart and complet he checkout process and submit

  • Profound Logic Have you been wondering about Node.js? Our free Node.js Webinar Series takes you from total beginner to creating a fully-functional IBM i Node.js business application.

  • SB Profound WC 5536Join us for this hour-long webcast that will explore:

  • Fortra IT managers hoping to find new IBM i talent are discovering that the pool of experienced RPG programmers and operators or administrators with intimate knowledge of the operating system and the applications that run on it is small. This begs the question: How will you manage the platform that supports such a big part of your business? This guide offers strategies and software suggestions to help you plan IT staffing and resources and smooth the transition after your AS/400 talent retires. Read on to learn: