non vorrei lavorare

昔はおもにプログラミングやガジェット系、今は?

mrubyのFile.openをちょっと調べたメモ

おはようございます。最近、兄弟で、意気投合して、結構な時間遊んで、部屋を散らかし放題にしています。@kjunichiです。

背景

iOSでmrubyをCordovaのプラグインとして動かしていて、mruby-ioを加えて、ビルドしたが、 cordovaのカメラプラグインで撮影した画像ファイルのパスをmrubyで開こうとしたら、アプリが落ちた。

mrubyでのFile#openの詳細を知りたくなった。

解析結果

  • file.rbにはopenが定義されていない。
  • file.cにもopenは定義されていない。

困った。

  • FileクラスはIOクラスを継承していることを知る
class File < IO
  • IO.openはio.rbで定義されていた!
  • IO.openの中身
io = self.new(*args)
  • なるほど、IO.newを呼んでいた

  • IO.newはCで定義されていた。()

mrb_define_method(mrb, io, "initialize", mrb_io_initialize, MRB_ARGS_ANY()); /* 15.2.20.5.21 (x)*/
  • mrb_io_initializeをみる
mrb_value
mrb_io_initialize(mrb_state *mrb, mrb_value io)
{
  struct mrb_io *fptr;
  mrb_int fd;
  mrb_value mode, opt;
  int flags;

  mode = opt = mrb_nil_value();

mrb_get_args(mrb, "i|So", &fd, &mode, &opt);

あれ?FD前提だ!?

mrb_get_args(mrb, "i|So", &fd, &mode, &opt);

rubyの引数パースの第一引数が整数型で決め打ちされてる!

Rubyオブジェクト指向

  • Fileクラスの場合、IO.newではなく、File.newが呼ばれる!
  • File.newを調べる
  • File.newはfile.rbで定義されていた
def initialize(fd_or_path, mode = "r", perm = 0666)
    if fd_or_path.kind_of? Fixnum
      super(fd_or_path, mode)
    else
      @path = fd_or_path
      fd = IO.sysopen(@path, mode, perm)
      super(fd, mode)
    end
end
fd = IO.sysopen(@path, mode, perm)
  • IO.sysopenを調べる
mrb_value
mrb_io_s_sysopen(mrb_state *mrb, mrb_value klass)
{
  mrb_value path = mrb_nil_value();
  mrb_value mode = mrb_nil_value();
  mrb_int fd, flags, perm = -1;
  const char *pat;
  int modenum;

  mrb_get_args(mrb, "S|Si", &path, &mode, &perm);
  if (mrb_nil_p(mode)) {
    mode = mrb_str_new_cstr(mrb, "r");
  }
  if (perm < 0) {
    perm = 0666;
  }

  pat = mrb_string_value_cstr(mrb, &path);
  flags = mrb_io_modestr_to_flags(mrb, mrb_string_value_cstr(mrb, &mode));
  modenum = mrb_io_flags_to_modenum(mrb, flags);
  fd = mrb_cloexec_open(mrb, pat, modenum, perm);
fd = mrb_cloexec_open(mrb, pat, modenum, perm);
  • mrb_cloexec_openを調べる
int
mrb_cloexec_open(mrb_state *mrb, const char *pathname, mrb_int flags, mrb_int mode)
{
  int fd, retry = FALSE;

#ifdef O_CLOEXEC
  /* O_CLOEXEC is available since Linux 2.6.23.  Linux 2.6.18 silently ignore it. */
  flags |= O_CLOEXEC;
#elif defined O_NOINHERIT
  flags |= O_NOINHERIT;
#endif
reopen:
fd = open(pathname, flags, mode);

やっとたどり着けた。

関連記事

11年前の記事

9年前の記事

5年前の記事

4年前の記事

2年前の記事