Swift runtime - calling superclass method












5















I'm creating a subclass of UIView in runtime and providing my implementation of the layoutSubviews method for it. One imporant thing that I need to do is to perform super.layoutSubviews(). In Objective-C I can do it using the objc_msgSendSuper function:



Class objectClass = object_getClass(object);
Class superclass = class_getSuperclass(objectClass);

struct objc_super superInfo;
superInfo.receiver = object;
superInfo.super_class = superclass;
typedef void *(*ObjCMsgSendSuperReturnVoid)(struct objc_super *, SEL);
ObjCMsgSendSuperReturnVoid sendMsgReturnVoid = (ObjCMsgSendSuperReturnVoid)objc_msgSendSuper;
sendMsgReturnVoid(&superInfo, @selector(layoutSubviews));


But objc_msgSendSuper method is unavailable in Swift. What should I use for performing the same thing?










share|improve this question





























    5















    I'm creating a subclass of UIView in runtime and providing my implementation of the layoutSubviews method for it. One imporant thing that I need to do is to perform super.layoutSubviews(). In Objective-C I can do it using the objc_msgSendSuper function:



    Class objectClass = object_getClass(object);
    Class superclass = class_getSuperclass(objectClass);

    struct objc_super superInfo;
    superInfo.receiver = object;
    superInfo.super_class = superclass;
    typedef void *(*ObjCMsgSendSuperReturnVoid)(struct objc_super *, SEL);
    ObjCMsgSendSuperReturnVoid sendMsgReturnVoid = (ObjCMsgSendSuperReturnVoid)objc_msgSendSuper;
    sendMsgReturnVoid(&superInfo, @selector(layoutSubviews));


    But objc_msgSendSuper method is unavailable in Swift. What should I use for performing the same thing?










    share|improve this question



























      5












      5








      5








      I'm creating a subclass of UIView in runtime and providing my implementation of the layoutSubviews method for it. One imporant thing that I need to do is to perform super.layoutSubviews(). In Objective-C I can do it using the objc_msgSendSuper function:



      Class objectClass = object_getClass(object);
      Class superclass = class_getSuperclass(objectClass);

      struct objc_super superInfo;
      superInfo.receiver = object;
      superInfo.super_class = superclass;
      typedef void *(*ObjCMsgSendSuperReturnVoid)(struct objc_super *, SEL);
      ObjCMsgSendSuperReturnVoid sendMsgReturnVoid = (ObjCMsgSendSuperReturnVoid)objc_msgSendSuper;
      sendMsgReturnVoid(&superInfo, @selector(layoutSubviews));


      But objc_msgSendSuper method is unavailable in Swift. What should I use for performing the same thing?










      share|improve this question
















      I'm creating a subclass of UIView in runtime and providing my implementation of the layoutSubviews method for it. One imporant thing that I need to do is to perform super.layoutSubviews(). In Objective-C I can do it using the objc_msgSendSuper function:



      Class objectClass = object_getClass(object);
      Class superclass = class_getSuperclass(objectClass);

      struct objc_super superInfo;
      superInfo.receiver = object;
      superInfo.super_class = superclass;
      typedef void *(*ObjCMsgSendSuperReturnVoid)(struct objc_super *, SEL);
      ObjCMsgSendSuperReturnVoid sendMsgReturnVoid = (ObjCMsgSendSuperReturnVoid)objc_msgSendSuper;
      sendMsgReturnVoid(&superInfo, @selector(layoutSubviews));


      But objc_msgSendSuper method is unavailable in Swift. What should I use for performing the same thing?







      swift objective-c-runtime






      share|improve this question















      share|improve this question













      share|improve this question




      share|improve this question








      edited Jan 2 at 12:29









      Hamish

      47.7k7109172




      47.7k7109172










      asked Jan 2 at 6:53









      Ruslan SerebriakovRuslan Serebriakov

      447310




      447310
























          1 Answer
          1






          active

          oldest

          votes


















          3














          As Martin says, objc_msgSendSuper isn't available in Swift because it's a C variadic function, which Swift doesn't import due to the lack of type safety.



          One alternative is to use class_getMethodImplementation in order to get a pointer to the function to call for a selector on a given class type. From there, you can cast it to a function type which Swift can call using unsafeBitCast, taking care that the parameter and return types match up.



          For example:



          import Foundation

          class C {
          @objc func foo() {
          print("C's foo")
          }
          }

          class D : C {
          override func foo() {
          print("D's foo")
          }
          }

          let d = D()

          let superclass: AnyClass = class_getSuperclass(type(of: d))!
          let selector = #selector(C.foo)

          // The function to call for a message send of "foo" to a `C` object.
          let impl = class_getMethodImplementation(superclass, selector)!

          // @convention(c) tells Swift this is a bare function pointer (with no context object)
          // All Obj-C method functions have the receiver and message as their first two parameters
          // Therefore this denotes a method of type `() -> Void`, which matches up with `foo`
          typealias ObjCVoidVoidFn = @convention(c) (AnyObject, Selector) -> Void

          let fn = unsafeBitCast(impl, to: ObjCVoidVoidFn.self)
          fn(d, selector) // C's foo


          Note that like objc_msgSendSuper this assumes that the return type bridged to Obj-C is layout compatible with a pointer. This is true in most cases (including yours), but wouldn't be true for a method returning a type such as CGRect, which is represented in Obj-C using a C structure type.



          For those cases, you would need to use class_getMethodImplementation_stret instead:



          import Foundation

          class C {
          @objc func bar() -> CGRect {
          return CGRect(x: 2, y: 3, width: 4, height: 5)
          }
          }

          class D : C {
          override func bar() -> CGRect {
          return .zero
          }
          }

          let d = D()

          let superclass: AnyClass = class_getSuperclass(type(of: d))!
          let selector = #selector(C.bar)
          let impl = class_getMethodImplementation_stret(superclass, selector)!

          typealias ObjCVoidVoidFn = @convention(c) (AnyObject, Selector) -> CGRect

          let fn = unsafeBitCast(impl, to: ObjCVoidVoidFn.self)
          let rect = fn(d, selector)
          print(rect) // (2.0, 3.0, 4.0, 5.0)


          The distinction between class_getMethodImplementation and class_getMethodImplementation_stret is due to the difference in calling convention – a word sized type can be passed back through a register, however a structure of larger size needs to be passed back indirectly. This matters for class_getMethodImplementation because it could pass back a thunk for message forwarding in the case where the object doesn't respond to the selector.



          Another option is to use method_getImplementation, which doesn't perform message forwarding and therefore doesn't need to distinguish between stret and non-stret.



          For example:



          let impl = method_getImplementation(class_getInstanceMethod(superclass, selector)!)


          However bear in mind that the documentation notes:




          class_getMethodImplementation may be faster than method_getImplementation(class_getInstanceMethod(cls, name)).







          share|improve this answer





















          • 1





            Just as an additional remark: objc_msgSend(Super) is not available in Swift because it takes a variable argument list. There may be workarounds for this problem, but your approach looks simpler and safer: you don't have to think about objc_msgSendSuper vs objc_msgSendSuper_stret vs objc_msgSendSuper_fpret ...

            – Martin R
            Jan 2 at 12:42













          • @MartinR Yup – although that being said, you would have to use class_getMethodImplementation_stret instead if the method being called returned a structure (this appears to be due to the fact that it could return a thunk for message forwarding in the case where the object doesn't respond to the selector). Though it looks like this distinction is made automatically if you use class_getMethodImplementation instead. I'll add a note to my answer regarding this.

            – Hamish
            Jan 2 at 13:40











          • I hadn't noticed that class_getMethodImplementation has those variants as well ...

            – Martin R
            Jan 2 at 13:57











          • @MartinR What's weird is that there's no class_getMethodImplementation_fpret ¯_(ツ)_/¯

            – Hamish
            Jan 2 at 14:12













          • (Unrelated to this question: Did you have a look at stackoverflow.com/q/54283053/1187415?)

            – Martin R
            Jan 23 at 10:31











          Your Answer






          StackExchange.ifUsing("editor", function () {
          StackExchange.using("externalEditor", function () {
          StackExchange.using("snippets", function () {
          StackExchange.snippets.init();
          });
          });
          }, "code-snippets");

          StackExchange.ready(function() {
          var channelOptions = {
          tags: "".split(" "),
          id: "1"
          };
          initTagRenderer("".split(" "), "".split(" "), channelOptions);

          StackExchange.using("externalEditor", function() {
          // Have to fire editor after snippets, if snippets enabled
          if (StackExchange.settings.snippets.snippetsEnabled) {
          StackExchange.using("snippets", function() {
          createEditor();
          });
          }
          else {
          createEditor();
          }
          });

          function createEditor() {
          StackExchange.prepareEditor({
          heartbeatType: 'answer',
          autoActivateHeartbeat: false,
          convertImagesToLinks: true,
          noModals: true,
          showLowRepImageUploadWarning: true,
          reputationToPostImages: 10,
          bindNavPrevention: true,
          postfix: "",
          imageUploader: {
          brandingHtml: "Powered by u003ca class="icon-imgur-white" href="https://imgur.com/"u003eu003c/au003e",
          contentPolicyHtml: "User contributions licensed under u003ca href="https://creativecommons.org/licenses/by-sa/3.0/"u003ecc by-sa 3.0 with attribution requiredu003c/au003e u003ca href="https://stackoverflow.com/legal/content-policy"u003e(content policy)u003c/au003e",
          allowUrls: true
          },
          onDemand: true,
          discardSelector: ".discard-answer"
          ,immediatelyShowMarkdownHelp:true
          });


          }
          });














          draft saved

          draft discarded


















          StackExchange.ready(
          function () {
          StackExchange.openid.initPostLogin('.new-post-login', 'https%3a%2f%2fstackoverflow.com%2fquestions%2f54002375%2fswift-runtime-calling-superclass-method%23new-answer', 'question_page');
          }
          );

          Post as a guest















          Required, but never shown

























          1 Answer
          1






          active

          oldest

          votes








          1 Answer
          1






          active

          oldest

          votes









          active

          oldest

          votes






          active

          oldest

          votes









          3














          As Martin says, objc_msgSendSuper isn't available in Swift because it's a C variadic function, which Swift doesn't import due to the lack of type safety.



          One alternative is to use class_getMethodImplementation in order to get a pointer to the function to call for a selector on a given class type. From there, you can cast it to a function type which Swift can call using unsafeBitCast, taking care that the parameter and return types match up.



          For example:



          import Foundation

          class C {
          @objc func foo() {
          print("C's foo")
          }
          }

          class D : C {
          override func foo() {
          print("D's foo")
          }
          }

          let d = D()

          let superclass: AnyClass = class_getSuperclass(type(of: d))!
          let selector = #selector(C.foo)

          // The function to call for a message send of "foo" to a `C` object.
          let impl = class_getMethodImplementation(superclass, selector)!

          // @convention(c) tells Swift this is a bare function pointer (with no context object)
          // All Obj-C method functions have the receiver and message as their first two parameters
          // Therefore this denotes a method of type `() -> Void`, which matches up with `foo`
          typealias ObjCVoidVoidFn = @convention(c) (AnyObject, Selector) -> Void

          let fn = unsafeBitCast(impl, to: ObjCVoidVoidFn.self)
          fn(d, selector) // C's foo


          Note that like objc_msgSendSuper this assumes that the return type bridged to Obj-C is layout compatible with a pointer. This is true in most cases (including yours), but wouldn't be true for a method returning a type such as CGRect, which is represented in Obj-C using a C structure type.



          For those cases, you would need to use class_getMethodImplementation_stret instead:



          import Foundation

          class C {
          @objc func bar() -> CGRect {
          return CGRect(x: 2, y: 3, width: 4, height: 5)
          }
          }

          class D : C {
          override func bar() -> CGRect {
          return .zero
          }
          }

          let d = D()

          let superclass: AnyClass = class_getSuperclass(type(of: d))!
          let selector = #selector(C.bar)
          let impl = class_getMethodImplementation_stret(superclass, selector)!

          typealias ObjCVoidVoidFn = @convention(c) (AnyObject, Selector) -> CGRect

          let fn = unsafeBitCast(impl, to: ObjCVoidVoidFn.self)
          let rect = fn(d, selector)
          print(rect) // (2.0, 3.0, 4.0, 5.0)


          The distinction between class_getMethodImplementation and class_getMethodImplementation_stret is due to the difference in calling convention – a word sized type can be passed back through a register, however a structure of larger size needs to be passed back indirectly. This matters for class_getMethodImplementation because it could pass back a thunk for message forwarding in the case where the object doesn't respond to the selector.



          Another option is to use method_getImplementation, which doesn't perform message forwarding and therefore doesn't need to distinguish between stret and non-stret.



          For example:



          let impl = method_getImplementation(class_getInstanceMethod(superclass, selector)!)


          However bear in mind that the documentation notes:




          class_getMethodImplementation may be faster than method_getImplementation(class_getInstanceMethod(cls, name)).







          share|improve this answer





















          • 1





            Just as an additional remark: objc_msgSend(Super) is not available in Swift because it takes a variable argument list. There may be workarounds for this problem, but your approach looks simpler and safer: you don't have to think about objc_msgSendSuper vs objc_msgSendSuper_stret vs objc_msgSendSuper_fpret ...

            – Martin R
            Jan 2 at 12:42













          • @MartinR Yup – although that being said, you would have to use class_getMethodImplementation_stret instead if the method being called returned a structure (this appears to be due to the fact that it could return a thunk for message forwarding in the case where the object doesn't respond to the selector). Though it looks like this distinction is made automatically if you use class_getMethodImplementation instead. I'll add a note to my answer regarding this.

            – Hamish
            Jan 2 at 13:40











          • I hadn't noticed that class_getMethodImplementation has those variants as well ...

            – Martin R
            Jan 2 at 13:57











          • @MartinR What's weird is that there's no class_getMethodImplementation_fpret ¯_(ツ)_/¯

            – Hamish
            Jan 2 at 14:12













          • (Unrelated to this question: Did you have a look at stackoverflow.com/q/54283053/1187415?)

            – Martin R
            Jan 23 at 10:31
















          3














          As Martin says, objc_msgSendSuper isn't available in Swift because it's a C variadic function, which Swift doesn't import due to the lack of type safety.



          One alternative is to use class_getMethodImplementation in order to get a pointer to the function to call for a selector on a given class type. From there, you can cast it to a function type which Swift can call using unsafeBitCast, taking care that the parameter and return types match up.



          For example:



          import Foundation

          class C {
          @objc func foo() {
          print("C's foo")
          }
          }

          class D : C {
          override func foo() {
          print("D's foo")
          }
          }

          let d = D()

          let superclass: AnyClass = class_getSuperclass(type(of: d))!
          let selector = #selector(C.foo)

          // The function to call for a message send of "foo" to a `C` object.
          let impl = class_getMethodImplementation(superclass, selector)!

          // @convention(c) tells Swift this is a bare function pointer (with no context object)
          // All Obj-C method functions have the receiver and message as their first two parameters
          // Therefore this denotes a method of type `() -> Void`, which matches up with `foo`
          typealias ObjCVoidVoidFn = @convention(c) (AnyObject, Selector) -> Void

          let fn = unsafeBitCast(impl, to: ObjCVoidVoidFn.self)
          fn(d, selector) // C's foo


          Note that like objc_msgSendSuper this assumes that the return type bridged to Obj-C is layout compatible with a pointer. This is true in most cases (including yours), but wouldn't be true for a method returning a type such as CGRect, which is represented in Obj-C using a C structure type.



          For those cases, you would need to use class_getMethodImplementation_stret instead:



          import Foundation

          class C {
          @objc func bar() -> CGRect {
          return CGRect(x: 2, y: 3, width: 4, height: 5)
          }
          }

          class D : C {
          override func bar() -> CGRect {
          return .zero
          }
          }

          let d = D()

          let superclass: AnyClass = class_getSuperclass(type(of: d))!
          let selector = #selector(C.bar)
          let impl = class_getMethodImplementation_stret(superclass, selector)!

          typealias ObjCVoidVoidFn = @convention(c) (AnyObject, Selector) -> CGRect

          let fn = unsafeBitCast(impl, to: ObjCVoidVoidFn.self)
          let rect = fn(d, selector)
          print(rect) // (2.0, 3.0, 4.0, 5.0)


          The distinction between class_getMethodImplementation and class_getMethodImplementation_stret is due to the difference in calling convention – a word sized type can be passed back through a register, however a structure of larger size needs to be passed back indirectly. This matters for class_getMethodImplementation because it could pass back a thunk for message forwarding in the case where the object doesn't respond to the selector.



          Another option is to use method_getImplementation, which doesn't perform message forwarding and therefore doesn't need to distinguish between stret and non-stret.



          For example:



          let impl = method_getImplementation(class_getInstanceMethod(superclass, selector)!)


          However bear in mind that the documentation notes:




          class_getMethodImplementation may be faster than method_getImplementation(class_getInstanceMethod(cls, name)).







          share|improve this answer





















          • 1





            Just as an additional remark: objc_msgSend(Super) is not available in Swift because it takes a variable argument list. There may be workarounds for this problem, but your approach looks simpler and safer: you don't have to think about objc_msgSendSuper vs objc_msgSendSuper_stret vs objc_msgSendSuper_fpret ...

            – Martin R
            Jan 2 at 12:42













          • @MartinR Yup – although that being said, you would have to use class_getMethodImplementation_stret instead if the method being called returned a structure (this appears to be due to the fact that it could return a thunk for message forwarding in the case where the object doesn't respond to the selector). Though it looks like this distinction is made automatically if you use class_getMethodImplementation instead. I'll add a note to my answer regarding this.

            – Hamish
            Jan 2 at 13:40











          • I hadn't noticed that class_getMethodImplementation has those variants as well ...

            – Martin R
            Jan 2 at 13:57











          • @MartinR What's weird is that there's no class_getMethodImplementation_fpret ¯_(ツ)_/¯

            – Hamish
            Jan 2 at 14:12













          • (Unrelated to this question: Did you have a look at stackoverflow.com/q/54283053/1187415?)

            – Martin R
            Jan 23 at 10:31














          3












          3








          3







          As Martin says, objc_msgSendSuper isn't available in Swift because it's a C variadic function, which Swift doesn't import due to the lack of type safety.



          One alternative is to use class_getMethodImplementation in order to get a pointer to the function to call for a selector on a given class type. From there, you can cast it to a function type which Swift can call using unsafeBitCast, taking care that the parameter and return types match up.



          For example:



          import Foundation

          class C {
          @objc func foo() {
          print("C's foo")
          }
          }

          class D : C {
          override func foo() {
          print("D's foo")
          }
          }

          let d = D()

          let superclass: AnyClass = class_getSuperclass(type(of: d))!
          let selector = #selector(C.foo)

          // The function to call for a message send of "foo" to a `C` object.
          let impl = class_getMethodImplementation(superclass, selector)!

          // @convention(c) tells Swift this is a bare function pointer (with no context object)
          // All Obj-C method functions have the receiver and message as their first two parameters
          // Therefore this denotes a method of type `() -> Void`, which matches up with `foo`
          typealias ObjCVoidVoidFn = @convention(c) (AnyObject, Selector) -> Void

          let fn = unsafeBitCast(impl, to: ObjCVoidVoidFn.self)
          fn(d, selector) // C's foo


          Note that like objc_msgSendSuper this assumes that the return type bridged to Obj-C is layout compatible with a pointer. This is true in most cases (including yours), but wouldn't be true for a method returning a type such as CGRect, which is represented in Obj-C using a C structure type.



          For those cases, you would need to use class_getMethodImplementation_stret instead:



          import Foundation

          class C {
          @objc func bar() -> CGRect {
          return CGRect(x: 2, y: 3, width: 4, height: 5)
          }
          }

          class D : C {
          override func bar() -> CGRect {
          return .zero
          }
          }

          let d = D()

          let superclass: AnyClass = class_getSuperclass(type(of: d))!
          let selector = #selector(C.bar)
          let impl = class_getMethodImplementation_stret(superclass, selector)!

          typealias ObjCVoidVoidFn = @convention(c) (AnyObject, Selector) -> CGRect

          let fn = unsafeBitCast(impl, to: ObjCVoidVoidFn.self)
          let rect = fn(d, selector)
          print(rect) // (2.0, 3.0, 4.0, 5.0)


          The distinction between class_getMethodImplementation and class_getMethodImplementation_stret is due to the difference in calling convention – a word sized type can be passed back through a register, however a structure of larger size needs to be passed back indirectly. This matters for class_getMethodImplementation because it could pass back a thunk for message forwarding in the case where the object doesn't respond to the selector.



          Another option is to use method_getImplementation, which doesn't perform message forwarding and therefore doesn't need to distinguish between stret and non-stret.



          For example:



          let impl = method_getImplementation(class_getInstanceMethod(superclass, selector)!)


          However bear in mind that the documentation notes:




          class_getMethodImplementation may be faster than method_getImplementation(class_getInstanceMethod(cls, name)).







          share|improve this answer















          As Martin says, objc_msgSendSuper isn't available in Swift because it's a C variadic function, which Swift doesn't import due to the lack of type safety.



          One alternative is to use class_getMethodImplementation in order to get a pointer to the function to call for a selector on a given class type. From there, you can cast it to a function type which Swift can call using unsafeBitCast, taking care that the parameter and return types match up.



          For example:



          import Foundation

          class C {
          @objc func foo() {
          print("C's foo")
          }
          }

          class D : C {
          override func foo() {
          print("D's foo")
          }
          }

          let d = D()

          let superclass: AnyClass = class_getSuperclass(type(of: d))!
          let selector = #selector(C.foo)

          // The function to call for a message send of "foo" to a `C` object.
          let impl = class_getMethodImplementation(superclass, selector)!

          // @convention(c) tells Swift this is a bare function pointer (with no context object)
          // All Obj-C method functions have the receiver and message as their first two parameters
          // Therefore this denotes a method of type `() -> Void`, which matches up with `foo`
          typealias ObjCVoidVoidFn = @convention(c) (AnyObject, Selector) -> Void

          let fn = unsafeBitCast(impl, to: ObjCVoidVoidFn.self)
          fn(d, selector) // C's foo


          Note that like objc_msgSendSuper this assumes that the return type bridged to Obj-C is layout compatible with a pointer. This is true in most cases (including yours), but wouldn't be true for a method returning a type such as CGRect, which is represented in Obj-C using a C structure type.



          For those cases, you would need to use class_getMethodImplementation_stret instead:



          import Foundation

          class C {
          @objc func bar() -> CGRect {
          return CGRect(x: 2, y: 3, width: 4, height: 5)
          }
          }

          class D : C {
          override func bar() -> CGRect {
          return .zero
          }
          }

          let d = D()

          let superclass: AnyClass = class_getSuperclass(type(of: d))!
          let selector = #selector(C.bar)
          let impl = class_getMethodImplementation_stret(superclass, selector)!

          typealias ObjCVoidVoidFn = @convention(c) (AnyObject, Selector) -> CGRect

          let fn = unsafeBitCast(impl, to: ObjCVoidVoidFn.self)
          let rect = fn(d, selector)
          print(rect) // (2.0, 3.0, 4.0, 5.0)


          The distinction between class_getMethodImplementation and class_getMethodImplementation_stret is due to the difference in calling convention – a word sized type can be passed back through a register, however a structure of larger size needs to be passed back indirectly. This matters for class_getMethodImplementation because it could pass back a thunk for message forwarding in the case where the object doesn't respond to the selector.



          Another option is to use method_getImplementation, which doesn't perform message forwarding and therefore doesn't need to distinguish between stret and non-stret.



          For example:



          let impl = method_getImplementation(class_getInstanceMethod(superclass, selector)!)


          However bear in mind that the documentation notes:




          class_getMethodImplementation may be faster than method_getImplementation(class_getInstanceMethod(cls, name)).








          share|improve this answer














          share|improve this answer



          share|improve this answer








          edited Jan 2 at 15:04

























          answered Jan 2 at 12:26









          HamishHamish

          47.7k7109172




          47.7k7109172








          • 1





            Just as an additional remark: objc_msgSend(Super) is not available in Swift because it takes a variable argument list. There may be workarounds for this problem, but your approach looks simpler and safer: you don't have to think about objc_msgSendSuper vs objc_msgSendSuper_stret vs objc_msgSendSuper_fpret ...

            – Martin R
            Jan 2 at 12:42













          • @MartinR Yup – although that being said, you would have to use class_getMethodImplementation_stret instead if the method being called returned a structure (this appears to be due to the fact that it could return a thunk for message forwarding in the case where the object doesn't respond to the selector). Though it looks like this distinction is made automatically if you use class_getMethodImplementation instead. I'll add a note to my answer regarding this.

            – Hamish
            Jan 2 at 13:40











          • I hadn't noticed that class_getMethodImplementation has those variants as well ...

            – Martin R
            Jan 2 at 13:57











          • @MartinR What's weird is that there's no class_getMethodImplementation_fpret ¯_(ツ)_/¯

            – Hamish
            Jan 2 at 14:12













          • (Unrelated to this question: Did you have a look at stackoverflow.com/q/54283053/1187415?)

            – Martin R
            Jan 23 at 10:31














          • 1





            Just as an additional remark: objc_msgSend(Super) is not available in Swift because it takes a variable argument list. There may be workarounds for this problem, but your approach looks simpler and safer: you don't have to think about objc_msgSendSuper vs objc_msgSendSuper_stret vs objc_msgSendSuper_fpret ...

            – Martin R
            Jan 2 at 12:42













          • @MartinR Yup – although that being said, you would have to use class_getMethodImplementation_stret instead if the method being called returned a structure (this appears to be due to the fact that it could return a thunk for message forwarding in the case where the object doesn't respond to the selector). Though it looks like this distinction is made automatically if you use class_getMethodImplementation instead. I'll add a note to my answer regarding this.

            – Hamish
            Jan 2 at 13:40











          • I hadn't noticed that class_getMethodImplementation has those variants as well ...

            – Martin R
            Jan 2 at 13:57











          • @MartinR What's weird is that there's no class_getMethodImplementation_fpret ¯_(ツ)_/¯

            – Hamish
            Jan 2 at 14:12













          • (Unrelated to this question: Did you have a look at stackoverflow.com/q/54283053/1187415?)

            – Martin R
            Jan 23 at 10:31








          1




          1





          Just as an additional remark: objc_msgSend(Super) is not available in Swift because it takes a variable argument list. There may be workarounds for this problem, but your approach looks simpler and safer: you don't have to think about objc_msgSendSuper vs objc_msgSendSuper_stret vs objc_msgSendSuper_fpret ...

          – Martin R
          Jan 2 at 12:42







          Just as an additional remark: objc_msgSend(Super) is not available in Swift because it takes a variable argument list. There may be workarounds for this problem, but your approach looks simpler and safer: you don't have to think about objc_msgSendSuper vs objc_msgSendSuper_stret vs objc_msgSendSuper_fpret ...

          – Martin R
          Jan 2 at 12:42















          @MartinR Yup – although that being said, you would have to use class_getMethodImplementation_stret instead if the method being called returned a structure (this appears to be due to the fact that it could return a thunk for message forwarding in the case where the object doesn't respond to the selector). Though it looks like this distinction is made automatically if you use class_getMethodImplementation instead. I'll add a note to my answer regarding this.

          – Hamish
          Jan 2 at 13:40





          @MartinR Yup – although that being said, you would have to use class_getMethodImplementation_stret instead if the method being called returned a structure (this appears to be due to the fact that it could return a thunk for message forwarding in the case where the object doesn't respond to the selector). Though it looks like this distinction is made automatically if you use class_getMethodImplementation instead. I'll add a note to my answer regarding this.

          – Hamish
          Jan 2 at 13:40













          I hadn't noticed that class_getMethodImplementation has those variants as well ...

          – Martin R
          Jan 2 at 13:57





          I hadn't noticed that class_getMethodImplementation has those variants as well ...

          – Martin R
          Jan 2 at 13:57













          @MartinR What's weird is that there's no class_getMethodImplementation_fpret ¯_(ツ)_/¯

          – Hamish
          Jan 2 at 14:12







          @MartinR What's weird is that there's no class_getMethodImplementation_fpret ¯_(ツ)_/¯

          – Hamish
          Jan 2 at 14:12















          (Unrelated to this question: Did you have a look at stackoverflow.com/q/54283053/1187415?)

          – Martin R
          Jan 23 at 10:31





          (Unrelated to this question: Did you have a look at stackoverflow.com/q/54283053/1187415?)

          – Martin R
          Jan 23 at 10:31




















          draft saved

          draft discarded




















































          Thanks for contributing an answer to Stack Overflow!


          • Please be sure to answer the question. Provide details and share your research!

          But avoid



          • Asking for help, clarification, or responding to other answers.

          • Making statements based on opinion; back them up with references or personal experience.


          To learn more, see our tips on writing great answers.




          draft saved


          draft discarded














          StackExchange.ready(
          function () {
          StackExchange.openid.initPostLogin('.new-post-login', 'https%3a%2f%2fstackoverflow.com%2fquestions%2f54002375%2fswift-runtime-calling-superclass-method%23new-answer', 'question_page');
          }
          );

          Post as a guest















          Required, but never shown





















































          Required, but never shown














          Required, but never shown












          Required, but never shown







          Required, but never shown

































          Required, but never shown














          Required, but never shown












          Required, but never shown







          Required, but never shown







          Popular posts from this blog

          MongoDB - Not Authorized To Execute Command

          in spring boot 2.1 many test slices are not allowed anymore due to multiple @BootstrapWith

          How to fix TextFormField cause rebuild widget in Flutter