Forwarding Methods Calls

Chapel 1.15 includes a preview version of a method forwarding feature. This feature allows a record or class to specify that certain method calls will be forwarded to a particular expression. The most typical use case is to forward methods to a particular field.

Why Forwarding?

While Chapel supports inheritance between classes as a way to reuse method implementations, inheritance is not always appropriate. In particular, inheritance has the drawback that it affects the types of the classes and therefore changes what users can do with those classes. Due to some of these drawbacks, there is a well-known design pattern composition over inheritance.

The basic strategy for using composition instead of inheritance is to declare a member field that stores the type that would be inherited from. Then, declare methods that forward to the member field. This can allow the implementation to be more flexible and re-usable. In particular, changing how many classes participate in the implementation of a particular type need not change its public interface or impact the users of that type.

Besides these issues, composition is a more general strategy in Chapel due to the following language design decisions:

  • records don’t support inheritance.

  • multiple inheritance for classes is not supported.

  • inheriting a record from a class or a class from a record is not supported.

Composition can help in each of these cases where inheritance cannot apply.

Example of Forwarding

Suppose, for example, MyCircleImpl is a Chapel class, but the author of that type wishes to present it as a record (say, MyCircle). Such a division allows the type author to control what happens when the type is copied and to manage the deletion of the class instances.

Consider the following example:

class MyCircleImpl {
  var radius:real;
  proc area() {
    return pi*radius*radius;
  }
  proc circumference() {
    return 2.0*pi*radius;
  }
}

record MyCircle {
  var impl: MyCircleImpl;

  // forwarding methods
  proc area() {
    return impl.area();
  }
  proc circumference() {
    return impl.circumference();
  }
}

Since writing these forwarding methods is repetitive and boring, it’s a good candidate for automation. The Chapel language allows automating the creation of these forwarding methods by using the forwarding keyword. For example, the following is equivalent to the previous MyCircle record.

record MyCircle {
  forwarding var impl: MyCircleImpl;
}

The forwarding keyword instructs the compiler to forward method invocations on a MyCircle that would otherwise not resolve to the field impl.

Using forwarding

Besides the forwarding var style, the forwarding syntax can be applied to forward to a particular expression. For example, one might wish to generate a different error in the event that impl is nil:

record MyCircle {
  var impl: MyCircleImpl;

  proc getImplOrFail() {
    if impl == nil then
      halt("impl is nil");
    else
      return impl;
  }

  forwarding getImplOrFail();
}

var empty = new MyCircle(nil);
empty.area(); // halts with "impl is nil"

Another direction that a user of forwarding might go is that they might decide to only forward certain methods. For example, the following are equivalent ways to forward only the area() method, assuming that the only methods of MyCircleImpl are area() and circumference().

record MyCircle {
  var impl: MyCircleImpl;

  forwarding impl only area;
}
record MyCircle {
  var impl: MyCircleImpl;

  forwarding impl except circumference;
}

As with use, forwarding supports comma-separated only and except lists.

Additionally, note that multiple forwarding declarations can be specified.

Resolving Forwarded Methods

Forwarded methods resolve only when regular methods on a particular type do not resolve. For example, in the above cases, if MyCircle declared a area() method that could be called with no arguments, that method would be called in preference to the forwarded method on MyCircleImpl that is available. Other than that, the forwarded methods participate normally in the function resolution process. As with other ambiguous function declarations, ambiguity errors will be reported at the call site.

Note that special IO methods such as writeThis and readThis are not forwarded since the compiler generates these for the type by default. Similarly, a field accessor will not get forwarded if the class defines a conflicting field or method name. Consider the example below:

record ArrayWrapper {
  var array: [1..0] int;

  proc shape {
    return 1;
  }

  forwarding array;
}

The shape method above will resolve, meaning array.shape will not be forwarded.