MEX编译错误:未知的类型名称'char16_t'。

21

由于以下错误,我无法编译任何MATLAB MEX代码:

In file included from /Applications/MATLAB_R2013a.app/extern/include/mex.h:58:
In file included from /Applications/MATLAB_R2013a.app/extern/include/matrix.h:294:
/Applications/MATLAB_R2013a.app/extern/include/tmwtypes.h:819:9: error: unknown type name 'char16_t'
typedef char16_t CHAR16_T;

据我所知,我机器上唯一改变的是Xcode升级到了版本5.1 (5B130a)。

有没有临时解决方案可以在MATLAB中编译MEX代码?

[运行在OS 10.9.2上,使用基于LLVM 3.4svn的Apple LLVM版本5.1(clang-503.0.38)]

7个回答

21

默认情况下,升级后的 Clang 不会设置 char16_t,而 MATLAB 需要它。

快速解决方案

这适用于 C 或 C++ 代码,但需要在每个 mex 命令行上执行。

>> mex -Dchar16_t=uint16_t ...

以下其他解决方案将此定义放入mex配置文件中或启用C++11。

永久解决方案

选项:

  • 在你的mex配置文件中添加-std=c++11CXXFLAGS,并编译.cpp文件而不是.c文件。 mex配置文件为mexopts.sh(R2014a之前)或由mex -setup指定的.xml文件(R2014a+)。 这是OP成功的解决方案,但下一个选项也可以使用。请确保编辑活动/安装配置,而不是系统范围的引用。如果您无法确定,请尝试下一个解决方案。
  • 使用#definetypedef在包含mex.h之前创建char16_t(请参见下面的“其他解决方法”)。
  • 在MATLAB的某个未来版本中,这将被修复。重新运行mex -setup让MATLAB为您重新配置它并使其正常工作。截至R2014a,这不起作用。
  • 作为最后的手段,您始终可以修改MATLAB安装,像Dennis建议一样修改MATLAB的tmwtypes.h,但我强烈建议不要修改MATLAB安装。

注意:如果您正在使用C语言并且不能或不想更改为C ++,请按照此其他答案中的解决方案操作,或查看下面的另一种解决方法。


其他解决方法

如果由于某些原因无法启用C++11标准,则可以使用预处理器定义char16_t。要么在#include "mex.h"之前放置#define char16_t uint16_t,要么使用编译器命令行设置它:

-Dchar16_t=uint16_t

或者,再包含 mex.h 之前使用 typedef

typedef uint16_t char16_t;

如果这些解决方案不起作用,可以尝试将uint16_t更改为UINT16_T。此外,有报道称仅包含uchar.h即可引入该类型,但其他人没有该头文件。


1
你应该更改CXXFLAGS,而不是CFLAGS,否则你会得到一个“错误:无效参数'-std=c++11'不允许使用'C/ObjC'”。 - YuppieNetworking
@flippex17 看看是否可以尝试更改 CXXFLAGS,如 YuppieNetworking所述。我不会修改twmtypes.h文件。很抱歉,我不知道修复软件本身的最佳方法是什么。 - chappjc
1
对我来说,当更改CXXFLAGS时,这个方法有效,但我的代码最初是C代码。我必须将其更改为C ++(重命名为.cpp并在Matlab中使用的相关函数之前添加 extern "C")。 - YuppieNetworking
1
@YuppieNetworking 很高兴听到它能正常工作 - 我无法测试。此外,你可以使用 clang -x c++ file.c 强制将 clang 编译为 C++ 而不更改扩展名。 - chappjc

8

我也遇到了同样的错误,是在升级到Xcode 5.1后发生的。

引起错误的文件tmwtypes.h中相关的代码行(818-824)如下:

#if defined(__STDC_UTF_16__) || (defined(_HAS_CHAR16_T_LANGUAGE_SUPPORT) && _HAS_CHAR16_T_LANGUAGE_SUPPORT)
typedef char16_t CHAR16_T;
#elif defined(_MSC_VER)
typedef wchar_t CHAR16_T;
#else
typedef UINT16_T CHAR16_T;
#endif

一种解决方案是简单地更改该行。
typedef char16_t CHAR16_T;

转化为

typedef UINT16_T CHAR16_T;

我必须承认,我不知道这是否影响了mex文件的任何功能或行为,但至少我能够再次使用mex编译我的c文件。

谢谢你的帮助。但是我用了不同的方法。 - p0lAris
我必须这样做才能在Xcode中调试我的MEX代码。谢谢你。+1 - rayryeng

2

我刚刚添加了自己的实验(仅限C ++)。

#define char16_t uint16_t 

在mex文件的其他部分引起了一些问题。实际上,在我的mex文件之后,char16_t被正确地定义。通过跟踪包含文件的链,适当的类型char16_t设置在名为__config的文件中:

typedef __char16_t char16_t;

这也是从<algorithm>中包含的第一个文件。因此,技巧在于在mex.h之前包含algorithm
#include <algorithm>
#include "mex.h"

并且进行适当的设置,仍然以多平台方式执行,并且不更改构建配置中的任何内容。

这个修复方法对我有效!我尝试了chappjc的建议,最初并没有起作用......但是在我修改了mexopts.sh文件之后,我忘记重新通过mex-setup设置mex。一旦我这样做了,它终于起作用了,但是这个方法和chappjc的方法都可以正常工作。 - rayryeng

2

如果这种方法不起作用,请参阅其他答案。

我使用Homebrew将我的gcc/g++编译器升级到了4.8版本,即gcc-4.8g++-4.8

之后,我在mexopts.sh文件中更改了以下行:

CXXFLAGS="-fno-common -fexceptions -arch $ARCHS -isysroot $MW_SDKROOT -mmacosx-version-min=$MACOSX_DEPLOYMENT_TARGET -std=c++11"

在我的mexopts.sh文件中,这是第150行。我只添加了-std=c++11标志,我想这就是chappjc所说的意思。
编辑:这在chappjc的更新中有所涉及!

是的,那就是我的意思。感谢您的接受,也可以随意点赞。 ;) 我认为最明智的做法是不要修改MATLAB安装,尽管显然大多数人持不同意见。 - chappjc

0
作为 XCode 5.1.1 的一部分,char16_t__config 中被定义,并从 typeinfo 调用。
您可以添加。
#include <typeinfo> 

之前

#include "mex.h"

需要定义char16_t


0
在包含mex.h之前,先包含uchar.h...这样就可以正常工作了。另外,上面的答案(添加-std=c++11)只适用于c ++,而不是c。
#include <uchar.h>
#include "mex.h"

一些仅限于C语言的解决方案,包括这个,可以在这里讨论(https://dev59.com/h33aa4cB1Zd3GeqPgapY#22705789)。 - chappjc

0
这篇文章可能会有所帮助:http://www.seaandsailor.com/matlab-xcode6.html 比我想象的要简单。只需将所有的10.x替换为您的OS X版本,并在mexopts.sh文件中的CLIBS中添加-Dchar16_t=UINT16_T
它适用于安装了Xcode 6的OS X 10.9 Mavericks。

网页内容由stack overflow 提供, 点击上面的
可以查看英文原文,
原文链接