Source – SV Verification Directory
Polymorphism
As the term ‘polymorphism’ implies, ‘poly’ means ‘many’ and ‘morph’ means ‘form’ or ‘shape’. A base class handle can invoke methods of its child class which has the same name. Hence, an object can take many forms.
- An extended class object can be assigned to the base class but not the other way round.
- Virtual Keyword should be used in parent class to override the method by the child class.
Example Code
// Parent class
class parent_class;
// class properties or variables
int a, b, s;
// task method
task func1 (int a, b);
s = a + b;
$display("Non-Virtual: Parent Class => a = %0d , b = %0d, sum s = %0d", a, b, s);
endtask
virtual task func2 (int a, b);
s = a + b;
$display("Virtual: Parent Class => a = %0d , b = %0d, sum s = %0d", a, b, s);
endtask
endclass
// Child Class
class child_class extends parent_class;
// class properties or variables
int a, b, m;
// task method
task func1 (int a, b);
m = a * b;
$display("Non-Virtual: Child Class => a = %0d , b = %0d, mul m = %0d", a, b, m);
endtask
virtual task func2 (int a, b); // good practice to declare a class method as virtual
m = a * b;
$display("Virtual: Child Class => a = %0d , b = %0d, mul m = %0d", a, b, m);
endtask
endclass
module tb_polymorphism;
parent_class pr_obj; // declare a handle for parent class
child_class cl_obj; // declare a handle for child class
task run(parent_class arg, int a, b);
arg.func1(a, b); // commented and uncomment to
arg.func2(a, b); // see the effects
endtask
initial begin
cl_obj = new( ); // construct an object
pr_obj = new( ); // WHEN THE CHILD CLASS HANDLE POINTS TO THE PARENT CLASS HANDLE
$display("---------------------- Polymorphism ----------------------");
run(pr_obj, 30, 10);
run(cl_obj, 30, 10);
end
endmodule
Execute the code in EDA Playground
In the example, the task run
behaves differently each time it is called, depending on the argument passed to it. This behavior is known as polymorphism, and it is achieved by combining the behaviour of the class pointer and inheritance. When the task run
is called, an argument is created, and based on this argument, the task’s pointer can reference either the base or the child class. This works because the base and child classes are of the same type, meaning the child class inherits all the characteristics of the base class. However, a potential issue arises when both the base and child classes define a method with the same name. In the example above, both classes contain func1
and func2
. When executing the child class, func1
from the base class is executed instead of the child’s implementation. This behavior defeats the purpose of polymorphism. To ensure that the child class’s method is executed, the virtual
keyword is prefixed to the function declaration. In the example, the function func2
executes as intended, demonstrating proper polymorphism.
Note: It is not permissible to assign a parent class handle to a child class handle directly. To address this issue, Dynamic Casting is employed.