+ __current->k_status = errno;
+ return SYSCALL_ESTATUS(errno);
+}
+
+__DEFINE_LXSYSCALL1(int, rmdir, const char*, pathname)
+{
+ int errno;
+ struct v_dnode* dnode;
+ if ((errno = vfs_walk(NULL, pathname, &dnode, NULL, 0))) {
+ goto done;
+ }
+ if ((dnode->super_block->fs->types & FSTYPE_ROFS)) {
+ errno = EROFS;
+ goto done;
+ }
+
+ if (dnode->inode->open_count) {
+ errno = EBUSY;
+ goto done;
+ }
+
+ if (dnode->inode->itype == VFS_INODE_TYPE_DIR) {
+ errno = dnode->inode->ops.rmdir(dnode->inode);
+ } else {
+ errno = ENOTDIR;
+ }
+
+done:
+ __current->k_status = errno;
+ return SYSCALL_ESTATUS(errno);
+}
+
+int
+__vfs_do_unlink(struct v_inode* inode)
+{
+ int errno;
+ if (inode->open_count) {
+ errno = EBUSY;
+ } else if (inode->itype != VFS_INODE_TYPE_DIR) {
+ // TODO handle symbolic link and type other than regular file
+ errno = inode->ops.unlink(inode);
+ } else {
+ errno = EISDIR;
+ }
+
+ return errno;
+}
+
+__DEFINE_LXSYSCALL1(int, unlink, const char*, pathname)
+{
+ int errno;
+ struct v_dnode* dnode;
+ if ((errno = vfs_walk(NULL, pathname, &dnode, NULL, 0))) {
+ goto done;
+ }
+ if ((dnode->super_block->fs->types & FSTYPE_ROFS)) {
+ errno = EROFS;
+ goto done;
+ }
+
+ errno = __vfs_do_unlink(dnode->inode);
+
+done:
+ __current->k_status = errno;
+ return SYSCALL_ESTATUS(errno);
+}
+
+__DEFINE_LXSYSCALL2(int, unlinkat, int, fd, const char*, pathname)
+{
+ int errno;
+ struct v_fd* fd_s;
+ if (!GET_FD(fd, fd_s)) {
+ errno = EBADF;
+ } else {
+ struct v_dnode* dnode;
+ if (!(errno = vfs_walk(fd_s->file->dnode, pathname, &dnode, NULL, 0))) {
+ errno = __vfs_do_unlink(dnode->inode);
+ }
+ }
+
+done: