长话短说
子类正在父类(super class) 范围内重新实现(重新定义)父类(super class)(基类)的虚函数,因为动态加载器要求它这样做。这对我来说没有任何意义。
示例:
class IO80211Controller : public IOEthernetController
{
virtual IOReturn enablePacketTimestamping(); // Implemented in binary, I can see the disassembly.
};
// .cpp - Redefinition with superclass namespace.
IOReturn IO80211Controller::enablePacketTimestamping()
{
return kIOReturnUnsupported; // This is from the disassembly of IO80211Controller
}
以上不是真正的 header ,我希望它接近它应该的样子——没有可用的 header 。
// .hpp
class AirPortBrcm4331 : public IO80211Controller
{
// Subclass stuff goes here
};
// .cpp - Redefinition with superclass namespace.
IOReturn IO80211Controller::enablePacketTimestamping()
{
return kIOReturnUnsupported; // This is from the disassembly of AirPortBrcm4331
}
背景
我正在研究 IO80211Family.kext(没有可用的 header ),特别是 IO80211Controller 类 - 我正在反转 header ,因此它会可以继承此类并创建自定义 802.11 驱动程序。
发现问题
IO80211Controller 定义了许多虚拟成员函数,我需要在我的反向头文件中声明它们。我创建了一个包含所有虚函数的头文件(从 IO80211Controller 的 vtable 中提取)并将其用于我的子类。
加载我的新 kext(带有子类)时,出现链接错误:
kxld[com.osxkernel.MyWirelessDriver]: The following symbols are unresolved for this kext: kxld[com.osxkernel.MyWirelessDriver]: IO80211Controller::enableFeature(IO80211FeatureCode, void*) kxld[com.osxkernel.MyWirelessDriver]: IO80211Controller::flowIdSupported() kxld[com.osxkernel.MyWirelessDriver]: IO80211Controller::apple80211_ioctl(IO80211Interface*, __ifnet*, unsigned long, void*) kxld[com.osxkernel.MyWirelessDriver]: IO80211Controller::enablePacketTimestamping() kxld[com.osxkernel.MyWirelessDriver]: IO80211Controller::hardwareOutputQueueDepth(IO80211Interface*) kxld[com.osxkernel.MyWirelessDriver]: IO80211Controller::disablePacketTimestamping() kxld[com.osxkernel.MyWirelessDriver]: IO80211Controller::performCountryCodeOperation(IO80211Interface*, IO80211CountryCodeOp) kxld[com.osxkernel.MyWirelessDriver]: IO80211Controller::requiresExplicitMBufRelease() kxld[com.osxkernel.MyWirelessDriver]: IO80211Controller::_RESERVEDIO80211Controllerless7() kxld[com.osxkernel.MyWirelessDriver]: IO80211Controller::stopDMA() Link failed (error code 5).
父类(super class)的反向 header 包含 50 多个虚拟成员函数,因此如果有任何链接问题,我会假设它是全有或全无。当向这些函数添加一个简单的实现(使用父类(super class)命名空间)时,链接错误就消失了。
出现两个问题
假设
我无法回答第一个问题,但我已经开始研究第二个问题。
我查看了 IO80211Family mach-o 符号表,所有有链接错误的函数在它们的类型字段中都不包含 N_EXT 位——这意味着它们不是外部的符号,而其他函数确实包含 N_EXT 位。
我不确定这会如何影响加载 kext 过程,所以我深入研究 XNU 源代码并寻找 kext 加载代码。这里有一个主要参与者称为 vtable 修补,这可能会阐明我的第一个问题。
不管怎样,有一个名为 kxld_sym_is_unresolved 的谓词函数可以检查符号是否未解析。 kxld 对所有符号调用此函数,以验证它们是否正常。
boolean_t
kxld_sym_is_unresolved(const KXLDSym *sym)
{
return ((kxld_sym_is_undefined(sym) && !kxld_sym_is_replaced(sym)) ||
kxld_sym_is_indirect(sym) || kxld_sym_is_common(sym));
}
在我的例子中,这个函数的结果归结为 kxld_sym_is_replaced 的返回值,它只是检查符号是否已被修补(vtable 修补),我不太了解它是什么并且它如何影响我...
大问题
为什么苹果选择这些功能不外挂?他们是否暗示他们应该由其他人实现——以及其他人,为什么与父类(super class)具有相同的范围?我跳进了源代码以找到答案,但没有找到。这是最让我不安的——它不符合我的逻辑。我知道一个完整的综合答案可能太复杂了,所以至少可以帮助我在更高层次上理解这里发生了什么,不让子类以这种奇怪的方式实现这些特定功能背后的逻辑是什么(为什么不是纯抽象)?
非常感谢您阅读本文!
最佳答案
直接的解释确实是符号不是由 IO80211 kext 导出的。然而,这背后的可能原因是函数是内联实现的,如下所示:
class IO80211Controller : public IOEthernetController
{
//...
virtual IOReturn enablePacketTimestamping()
{
return kIOReturnUnsupported;
}
//...
};
例如,如果我构建这段代码:
#include <cstdio>
class MyClass
{
public:
virtual void InlineVirtual() { printf("MyClass::InlineVirtual\n"); }
virtual void RegularVirtual();
};
void MyClass::RegularVirtual()
{
printf("MyClass::RegularVirtual\n");
}
int main()
{
MyClass a;
a.InlineVirtual();
a.RegularVirtual();
}
使用命令
clang++ -std=gnu++14 inline-virtual.cpp -o inline-virtual
然后使用 nm 检查符号:
$ nm ./inline-virtual
0000000100000f10 t __ZN7MyClass13InlineVirtualEv
0000000100000e90 T __ZN7MyClass14RegularVirtualEv
0000000100000ef0 t __ZN7MyClassC1Ev
0000000100000f40 t __ZN7MyClassC2Ev
0000000100001038 S __ZTI7MyClass
0000000100000faf S __ZTS7MyClass
0000000100001018 S __ZTV7MyClass
U __ZTVN10__cxxabiv117__class_type_infoE
0000000100000000 T __mh_execute_header
0000000100000ec0 T _main
U _printf
U dyld_stub_binder
可以看到MyClass::InlineVirtual隐藏了可见性(t),而MyClass::RegularVirtual被导出( T).必须在调用它的所有编译单元中提供声明为 inline 的函数的实现(显式地使用关键字或隐式地将其放置在 class 定义中),因此他们没有外部链接是有道理的。
关于c++ - 必须重新定义一些 kext 成员函数,以避免未解析的符号,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/52378165/
我正在尝试设置一个puppet节点,但rubygems似乎不正常。如果我通过它自己的二进制文件(/usr/lib/ruby/gems/1.8/gems/facter-1.5.8/bin/facter)在cli上运行facter,它工作正常,但如果我通过由rubygems(/usr/bin/facter)安装的二进制文件,它抛出:/usr/lib/ruby/1.8/facter/uptime.rb:11:undefinedmethod`get_uptime'forFacter::Util::Uptime:Module(NoMethodError)from/usr/lib/ruby
我在我的项目中添加了一个系统来重置用户密码并通过电子邮件将密码发送给他,以防他忘记密码。昨天它运行良好(当我实现它时)。当我今天尝试启动服务器时,出现以下错误。=>BootingWEBrick=>Rails3.2.1applicationstartingindevelopmentonhttp://0.0.0.0:3000=>Callwith-dtodetach=>Ctrl-CtoshutdownserverExiting/Users/vinayshenoy/.rvm/gems/ruby-1.9.3-p0/gems/actionmailer-3.2.1/lib/action_mailer
我的瘦服务器配置了nginx,我的ROR应用程序正在它们上运行。在我发布代码更新时运行thinrestart会给我的应用程序带来一些停机时间。我试图弄清楚如何优雅地重启正在运行的Thin实例,但找不到好的解决方案。有没有人能做到这一点? 最佳答案 #Restartjustthethinserverdescribedbythatconfigsudothin-C/etc/thin/mysite.ymlrestartNginx将继续运行并代理请求。如果您将Nginx设置为使用多个上游服务器,例如server{listen80;server
鉴于我有以下迁移:Sequel.migrationdoupdoalter_table:usersdoadd_column:is_admin,:default=>falseend#SequelrunsaDESCRIBEtablestatement,whenthemodelisloaded.#Atthispoint,itdoesnotknowthatusershaveais_adminflag.#Soitfails.@user=User.find(:email=>"admin@fancy-startup.example")@user.is_admin=true@user.save!ende
我想向我的Controller传递一个参数,它是一个简单的复选框,但我不知道如何在模型的form_for中引入它,这是我的观点:{:id=>'go_finance'}do|f|%>Transferirde:para:Entrada:"input",:placeholder=>"Quantofoiganho?"%>Saída:"output",:placeholder=>"Quantofoigasto?"%>Nota:我想做一个额外的复选框,但我该怎么做,模型中没有一个对象,而是一个要检查的对象,以便在Controller中创建一个ifelse,如果没有检查,请帮助我,非常感谢,谢谢
我已经从我的命令行中获得了一切,所以我可以运行rubymyfile并且它可以正常工作。但是当我尝试从sublime中运行它时,我得到了undefinedmethod`require_relative'formain:Object有人知道我的sublime设置中缺少什么吗?我正在使用OSX并安装了rvm。 最佳答案 或者,您可以只使用“require”,它应该可以正常工作。我认为“require_relative”仅适用于ruby1.9+ 关于ruby-主要:Objectwhenrun
我正在使用active_admin,我在Rails3应用程序的应用程序中有一个目录管理,其中包含模型和页面的声明。时不时地我也有一个类,当那个类有一个常量时,就像这样:classFooBAR="bar"end然后,我在每个必须在我的Rails应用程序中重新加载一些代码的请求中收到此警告:/Users/pupeno/helloworld/app/admin/billing.rb:12:warning:alreadyinitializedconstantBAR知道发生了什么以及如何避免这些警告吗? 最佳答案 在纯Ruby中:classA
我想在一个没有Sass引擎的类中使用Sass颜色函数。我已经在项目中使用了sassgem,所以我认为搭载会像以下一样简单:classRectangleincludeSass::Script::FunctionsdefcolorSass::Script::Color.new([0x82,0x39,0x06])enddefrender#hamlengineexecutedwithcontextofself#sothatwithintemlateicouldcall#%stop{offset:'0%',stop:{color:lighten(color)}}endend更新:参见上面的#re
我正在尝试用ruby中的gsub函数替换字符串中的某些单词,但有时效果很好,在某些情况下会出现此错误?这种格式有什么问题吗NoMethodError(undefinedmethod`gsub!'fornil:NilClass):模型.rbclassTest"replacethisID1",WAY=>"replacethisID2andID3",DELTA=>"replacethisID4"}end另一个模型.rbclassCheck 最佳答案 啊,我找到了!gsub!是一个非常奇怪的方法。首先,它替换了字符串,所以它实际上修改了
对于Rails模型,是否可以/建议让一个类的成员不持久保存到数据库中?我想将用户最后选择的类型存储在session变量中。由于我无法从我的模型中设置session变量,我想将值存储在一个“虚拟”类成员中,该成员只是将值传递回Controller。你能有这样的类(class)成员吗? 最佳答案 将非持久属性添加到Rails模型就像任何其他Ruby类一样:classUser扩展解释:在Ruby中,所有实例变量都是私有(private)的,不需要在赋值前定义。attr_accessor创建一个setter和getter方法:classUs