Changeset 26e7d6d in mainline for uspace/lib/fs/libfs.c
- Timestamp:
- 2011-09-19T16:31:00Z (13 years ago)
- Branches:
- lfn, master, serial, ticket/834-toolchain-update, topic/msim-upgrade, topic/simplify-dev-export
- Children:
- a347a11
- Parents:
- 3842a955 (diff), 086290d (diff)
Note: this is a merge changeset, the changes displayed below correspond to the merge itself.
Use the(diff)
links above to see all the changes relative to each parent. - File:
-
- 1 edited
Legend:
- Unmodified
- Added
- Removed
-
uspace/lib/fs/libfs.c
r3842a955 r26e7d6d 45 45 #include <mem.h> 46 46 #include <sys/stat.h> 47 #include <stdlib.h> 47 48 48 49 #define on_error(rc, action) \ … … 61 62 } while (0) 62 63 64 static fs_reg_t reg; 65 66 static vfs_out_ops_t *vfs_out_ops = NULL; 67 static libfs_ops_t *libfs_ops = NULL; 68 69 static void libfs_mount(libfs_ops_t *, fs_handle_t, ipc_callid_t, ipc_call_t *); 70 static void libfs_unmount(libfs_ops_t *, ipc_callid_t, ipc_call_t *); 71 static void libfs_lookup(libfs_ops_t *, fs_handle_t, ipc_callid_t, 72 ipc_call_t *); 73 static void libfs_stat(libfs_ops_t *, fs_handle_t, ipc_callid_t, ipc_call_t *); 74 static void libfs_open_node(libfs_ops_t *, fs_handle_t, ipc_callid_t, 75 ipc_call_t *); 76 77 static void vfs_out_mounted(ipc_callid_t rid, ipc_call_t *req) 78 { 79 service_id_t service_id = (service_id_t) IPC_GET_ARG1(*req); 80 char *opts; 81 int rc; 82 83 /* Accept the mount options. */ 84 rc = async_data_write_accept((void **) &opts, true, 0, 0, 0, NULL); 85 if (rc != EOK) { 86 async_answer_0(rid, rc); 87 return; 88 } 89 90 fs_index_t index; 91 aoff64_t size; 92 unsigned lnkcnt; 93 rc = vfs_out_ops->mounted(service_id, opts, &index, &size, &lnkcnt); 94 95 if (rc == EOK) 96 async_answer_4(rid, EOK, index, LOWER32(size), UPPER32(size), 97 lnkcnt); 98 else 99 async_answer_0(rid, rc); 100 101 free(opts); 102 } 103 104 static void vfs_out_mount(ipc_callid_t rid, ipc_call_t *req) 105 { 106 libfs_mount(libfs_ops, reg.fs_handle, rid, req); 107 } 108 109 static void vfs_out_unmounted(ipc_callid_t rid, ipc_call_t *req) 110 { 111 service_id_t service_id = (service_id_t) IPC_GET_ARG1(*req); 112 int rc; 113 114 rc = vfs_out_ops->unmounted(service_id); 115 116 async_answer_0(rid, rc); 117 } 118 119 static void vfs_out_unmount(ipc_callid_t rid, ipc_call_t *req) 120 { 121 122 libfs_unmount(libfs_ops, rid, req); 123 } 124 125 static void vfs_out_lookup(ipc_callid_t rid, ipc_call_t *req) 126 { 127 libfs_lookup(libfs_ops, reg.fs_handle, rid, req); 128 } 129 130 static void vfs_out_read(ipc_callid_t rid, ipc_call_t *req) 131 { 132 service_id_t service_id = (service_id_t) IPC_GET_ARG1(*req); 133 fs_index_t index = (fs_index_t) IPC_GET_ARG2(*req); 134 aoff64_t pos = (aoff64_t) MERGE_LOUP32(IPC_GET_ARG3(*req), 135 IPC_GET_ARG4(*req)); 136 size_t rbytes; 137 int rc; 138 139 rc = vfs_out_ops->read(service_id, index, pos, &rbytes); 140 141 if (rc == EOK) 142 async_answer_1(rid, EOK, rbytes); 143 else 144 async_answer_0(rid, rc); 145 } 146 147 static void vfs_out_write(ipc_callid_t rid, ipc_call_t *req) 148 { 149 service_id_t service_id = (service_id_t) IPC_GET_ARG1(*req); 150 fs_index_t index = (fs_index_t) IPC_GET_ARG2(*req); 151 aoff64_t pos = (aoff64_t) MERGE_LOUP32(IPC_GET_ARG3(*req), 152 IPC_GET_ARG4(*req)); 153 size_t wbytes; 154 aoff64_t nsize; 155 int rc; 156 157 rc = vfs_out_ops->write(service_id, index, pos, &wbytes, &nsize); 158 159 if (rc == EOK) 160 async_answer_3(rid, EOK, wbytes, LOWER32(nsize), UPPER32(nsize)); 161 else 162 async_answer_0(rid, rc); 163 } 164 165 static void vfs_out_truncate(ipc_callid_t rid, ipc_call_t *req) 166 { 167 service_id_t service_id = (service_id_t) IPC_GET_ARG1(*req); 168 fs_index_t index = (fs_index_t) IPC_GET_ARG2(*req); 169 aoff64_t size = (aoff64_t) MERGE_LOUP32(IPC_GET_ARG3(*req), 170 IPC_GET_ARG4(*req)); 171 int rc; 172 173 rc = vfs_out_ops->truncate(service_id, index, size); 174 175 async_answer_0(rid, rc); 176 } 177 178 static void vfs_out_close(ipc_callid_t rid, ipc_call_t *req) 179 { 180 service_id_t service_id = (service_id_t) IPC_GET_ARG1(*req); 181 fs_index_t index = (fs_index_t) IPC_GET_ARG2(*req); 182 int rc; 183 184 rc = vfs_out_ops->close(service_id, index); 185 186 async_answer_0(rid, rc); 187 } 188 189 static void vfs_out_destroy(ipc_callid_t rid, ipc_call_t *req) 190 { 191 service_id_t service_id = (service_id_t) IPC_GET_ARG1(*req); 192 fs_index_t index = (fs_index_t) IPC_GET_ARG2(*req); 193 int rc; 194 195 rc = vfs_out_ops->destroy(service_id, index); 196 197 async_answer_0(rid, rc); 198 } 199 200 static void vfs_out_open_node(ipc_callid_t rid, ipc_call_t *req) 201 { 202 libfs_open_node(libfs_ops, reg.fs_handle, rid, req); 203 } 204 205 static void vfs_out_stat(ipc_callid_t rid, ipc_call_t *req) 206 { 207 libfs_stat(libfs_ops, reg.fs_handle, rid, req); 208 } 209 210 static void vfs_out_sync(ipc_callid_t rid, ipc_call_t *req) 211 { 212 service_id_t service_id = (service_id_t) IPC_GET_ARG1(*req); 213 fs_index_t index = (fs_index_t) IPC_GET_ARG2(*req); 214 int rc; 215 216 rc = vfs_out_ops->sync(service_id, index); 217 218 async_answer_0(rid, rc); 219 } 220 221 static void vfs_connection(ipc_callid_t iid, ipc_call_t *icall, void *arg) 222 { 223 if (iid) { 224 /* 225 * This only happens for connections opened by 226 * IPC_M_CONNECT_ME_TO calls as opposed to callback connections 227 * created by IPC_M_CONNECT_TO_ME. 228 */ 229 async_answer_0(iid, EOK); 230 } 231 232 while (true) { 233 ipc_call_t call; 234 ipc_callid_t callid = async_get_call(&call); 235 236 if (!IPC_GET_IMETHOD(call)) 237 return; 238 239 switch (IPC_GET_IMETHOD(call)) { 240 case VFS_OUT_MOUNTED: 241 vfs_out_mounted(callid, &call); 242 break; 243 case VFS_OUT_MOUNT: 244 vfs_out_mount(callid, &call); 245 break; 246 case VFS_OUT_UNMOUNTED: 247 vfs_out_unmounted(callid, &call); 248 break; 249 case VFS_OUT_UNMOUNT: 250 vfs_out_unmount(callid, &call); 251 break; 252 case VFS_OUT_LOOKUP: 253 vfs_out_lookup(callid, &call); 254 break; 255 case VFS_OUT_READ: 256 vfs_out_read(callid, &call); 257 break; 258 case VFS_OUT_WRITE: 259 vfs_out_write(callid, &call); 260 break; 261 case VFS_OUT_TRUNCATE: 262 vfs_out_truncate(callid, &call); 263 break; 264 case VFS_OUT_CLOSE: 265 vfs_out_close(callid, &call); 266 break; 267 case VFS_OUT_DESTROY: 268 vfs_out_destroy(callid, &call); 269 break; 270 case VFS_OUT_OPEN_NODE: 271 vfs_out_open_node(callid, &call); 272 break; 273 case VFS_OUT_STAT: 274 vfs_out_stat(callid, &call); 275 break; 276 case VFS_OUT_SYNC: 277 vfs_out_sync(callid, &call); 278 break; 279 default: 280 async_answer_0(callid, ENOTSUP); 281 break; 282 } 283 } 284 } 285 63 286 /** Register file system server. 64 287 * … … 68 291 * 69 292 * @param sess Session for communication with VFS. 70 * @param reg File system registration structure. It will be71 * initialized by this function.72 293 * @param info VFS info structure supplied by the file system 73 294 * implementation. 74 * @param conn Connection fibril for handling all calls originating in75 * VFS.295 * @param vops Address of the vfs_out_ops_t structure. 296 * @param lops Address of the libfs_ops_t structure. 76 297 * 77 298 * @return EOK on success or a non-zero error code on errror. 78 299 * 79 300 */ 80 int fs_register(async_sess_t *sess, fs_reg_t *reg, vfs_info_t *info,81 async_client_conn_t conn)301 int fs_register(async_sess_t *sess, vfs_info_t *info, vfs_out_ops_t *vops, 302 libfs_ops_t *lops) 82 303 { 83 304 /* … … 104 325 105 326 /* 327 * Set VFS_OUT and libfs operations. 328 */ 329 vfs_out_ops = vops; 330 libfs_ops = lops; 331 332 /* 106 333 * Ask VFS for callback connection. 107 334 */ 108 async_connect_to_me(exch, 0, 0, 0, conn, NULL);335 async_connect_to_me(exch, 0, 0, 0, vfs_connection, NULL); 109 336 110 337 /* 111 338 * Allocate piece of address space for PLB. 112 339 */ 113 reg ->plb_ro = as_get_mappable_page(PLB_SIZE);114 if (!reg ->plb_ro) {340 reg.plb_ro = as_get_mappable_page(PLB_SIZE); 341 if (!reg.plb_ro) { 115 342 async_exchange_end(exch); 116 343 async_wait_for(req, NULL); … … 121 348 * Request sharing the Path Lookup Buffer with VFS. 122 349 */ 123 rc = async_share_in_start_0_0(exch, reg ->plb_ro, PLB_SIZE);350 rc = async_share_in_start_0_0(exch, reg.plb_ro, PLB_SIZE); 124 351 125 352 async_exchange_end(exch); … … 134 361 */ 135 362 async_wait_for(req, NULL); 136 reg ->fs_handle = (int) IPC_GET_ARG1(answer);363 reg.fs_handle = (int) IPC_GET_ARG1(answer); 137 364 138 365 /* … … 140 367 * the same connection fibril as well. 141 368 */ 142 async_set_client_connection( conn);369 async_set_client_connection(vfs_connection); 143 370 144 371 return IPC_GET_RETVAL(answer); … … 151 378 152 379 void libfs_mount(libfs_ops_t *ops, fs_handle_t fs_handle, ipc_callid_t rid, 153 ipc_call_t *req uest)154 { 155 devmap_handle_t mp_devmap_handle = (devmap_handle_t) IPC_GET_ARG1(*request);156 fs_index_t mp_fs_index = (fs_index_t) IPC_GET_ARG2(*req uest);157 fs_handle_t mr_fs_handle = (fs_handle_t) IPC_GET_ARG3(*req uest);158 devmap_handle_t mr_devmap_handle = (devmap_handle_t) IPC_GET_ARG4(*request);380 ipc_call_t *req) 381 { 382 service_id_t mp_service_id = (service_id_t) IPC_GET_ARG1(*req); 383 fs_index_t mp_fs_index = (fs_index_t) IPC_GET_ARG2(*req); 384 fs_handle_t mr_fs_handle = (fs_handle_t) IPC_GET_ARG3(*req); 385 service_id_t mr_service_id = (service_id_t) IPC_GET_ARG4(*req); 159 386 160 387 async_sess_t *mountee_sess = async_clone_receive(EXCHANGE_PARALLEL); … … 165 392 166 393 fs_node_t *fn; 167 int res = ops->node_get(&fn, mp_ devmap_handle, mp_fs_index);394 int res = ops->node_get(&fn, mp_service_id, mp_fs_index); 168 395 if ((res != EOK) || (!fn)) { 169 396 async_hangup(mountee_sess); … … 195 422 ipc_call_t answer; 196 423 int rc = async_data_write_forward_1_1(exch, VFS_OUT_MOUNTED, 197 mr_ devmap_handle, &answer);424 mr_service_id, &answer); 198 425 async_exchange_end(exch); 199 426 … … 201 428 fn->mp_data.mp_active = true; 202 429 fn->mp_data.fs_handle = mr_fs_handle; 203 fn->mp_data. devmap_handle = mr_devmap_handle;430 fn->mp_data.service_id = mr_service_id; 204 431 fn->mp_data.sess = mountee_sess; 205 432 } … … 208 435 * Do not release the FS node so that it stays in memory. 209 436 */ 210 async_answer_ 3(rid, rc, IPC_GET_ARG1(answer), IPC_GET_ARG2(answer),211 IPC_GET_ARG3(answer) );212 } 213 214 void libfs_unmount(libfs_ops_t *ops, ipc_callid_t rid, ipc_call_t *req uest)215 { 216 devmap_handle_t mp_devmap_handle = (devmap_handle_t) IPC_GET_ARG1(*request);217 fs_index_t mp_fs_index = (fs_index_t) IPC_GET_ARG2(*req uest);437 async_answer_4(rid, rc, IPC_GET_ARG1(answer), IPC_GET_ARG2(answer), 438 IPC_GET_ARG3(answer), IPC_GET_ARG4(answer)); 439 } 440 441 void libfs_unmount(libfs_ops_t *ops, ipc_callid_t rid, ipc_call_t *req) 442 { 443 service_id_t mp_service_id = (service_id_t) IPC_GET_ARG1(*req); 444 fs_index_t mp_fs_index = (fs_index_t) IPC_GET_ARG2(*req); 218 445 fs_node_t *fn; 219 446 int res; 220 447 221 res = ops->node_get(&fn, mp_ devmap_handle, mp_fs_index);448 res = ops->node_get(&fn, mp_service_id, mp_fs_index); 222 449 if ((res != EOK) || (!fn)) { 223 450 async_answer_0(rid, combine_rc(res, ENOENT)); … … 238 465 */ 239 466 async_exch_t *exch = async_exchange_begin(fn->mp_data.sess); 240 res = async_req_1_0(exch, VFS_OUT_UNMOUNTED, fn->mp_data. devmap_handle);467 res = async_req_1_0(exch, VFS_OUT_UNMOUNTED, fn->mp_data.service_id); 241 468 async_exchange_end(exch); 242 469 … … 248 475 fn->mp_data.mp_active = false; 249 476 fn->mp_data.fs_handle = 0; 250 fn->mp_data. devmap_handle= 0;477 fn->mp_data.service_id = 0; 251 478 fn->mp_data.sess = NULL; 252 479 … … 257 484 (void) ops->node_put(fn); 258 485 async_answer_0(rid, res); 486 } 487 488 static char plb_get_char(unsigned pos) 489 { 490 return reg.plb_ro[pos % PLB_SIZE]; 259 491 } 260 492 … … 273 505 */ 274 506 void libfs_lookup(libfs_ops_t *ops, fs_handle_t fs_handle, ipc_callid_t rid, 275 ipc_call_t *req uest)276 { 277 unsigned int first = IPC_GET_ARG1(*req uest);278 unsigned int last = IPC_GET_ARG2(*req uest);507 ipc_call_t *req) 508 { 509 unsigned int first = IPC_GET_ARG1(*req); 510 unsigned int last = IPC_GET_ARG2(*req); 279 511 unsigned int next = first; 280 devmap_handle_t devmap_handle = IPC_GET_ARG3(*request);281 int lflag = IPC_GET_ARG4(*req uest);282 fs_index_t index = IPC_GET_ARG5(*req uest);512 service_id_t service_id = IPC_GET_ARG3(*req); 513 int lflag = IPC_GET_ARG4(*req); 514 fs_index_t index = IPC_GET_ARG5(*req); 283 515 char component[NAME_MAX + 1]; 284 516 int len; … … 292 524 fs_node_t *tmp = NULL; 293 525 294 rc = ops->root_get(&cur, devmap_handle);526 rc = ops->root_get(&cur, service_id); 295 527 on_error(rc, goto out_with_answer); 296 528 … … 298 530 async_exch_t *exch = async_exchange_begin(cur->mp_data.sess); 299 531 async_forward_slow(rid, exch, VFS_OUT_LOOKUP, next, last, 300 cur->mp_data.devmap_handle, lflag, index, IPC_FF_ROUTE_FROM_ME); 532 cur->mp_data.service_id, lflag, index, 533 IPC_FF_ROUTE_FROM_ME); 301 534 async_exchange_end(exch); 302 535 … … 306 539 307 540 /* Eat slash */ 308 if ( ops->plb_get_char(next) == '/')541 if (plb_get_char(next) == '/') 309 542 next++; 310 543 … … 319 552 /* Collect the component */ 320 553 len = 0; 321 while ((next <= last) && ( ops->plb_get_char(next) != '/')) {554 while ((next <= last) && (plb_get_char(next) != '/')) { 322 555 if (len + 1 == NAME_MAX) { 323 556 /* Component length overflow */ … … 325 558 goto out; 326 559 } 327 component[len++] = ops->plb_get_char(next);560 component[len++] = plb_get_char(next); 328 561 /* Process next character */ 329 562 next++; … … 357 590 358 591 async_exch_t *exch = async_exchange_begin(tmp->mp_data.sess); 359 async_forward_slow(rid, exch, VFS_OUT_LOOKUP, next, last,360 tmp->mp_data.devmap_handle, lflag, index,592 async_forward_slow(rid, exch, VFS_OUT_LOOKUP, next, 593 last, tmp->mp_data.service_id, lflag, index, 361 594 IPC_FF_ROUTE_FROM_ME); 362 595 async_exchange_end(exch); … … 387 620 fs_node_t *fn; 388 621 if (lflag & L_CREATE) 389 rc = ops->create(&fn, devmap_handle,622 rc = ops->create(&fn, service_id, 390 623 lflag); 391 624 else 392 rc = ops->node_get(&fn, devmap_handle,625 rc = ops->node_get(&fn, service_id, 393 626 index); 394 627 on_error(rc, goto out_with_answer); … … 405 638 aoff64_t size = ops->size_get(fn); 406 639 async_answer_5(rid, fs_handle, 407 devmap_handle,640 service_id, 408 641 ops->index_get(fn), 409 642 LOWER32(size), … … 451 684 len = 0; 452 685 while (next <= last) { 453 if ( ops->plb_get_char(next) == '/') {686 if (plb_get_char(next) == '/') { 454 687 /* More than one component */ 455 688 async_answer_0(rid, ENOENT); … … 463 696 } 464 697 465 component[len++] = ops->plb_get_char(next);698 component[len++] = plb_get_char(next); 466 699 /* Process next character */ 467 700 next++; … … 473 706 fs_node_t *fn; 474 707 if (lflag & L_CREATE) 475 rc = ops->create(&fn, devmap_handle, lflag);708 rc = ops->create(&fn, service_id, lflag); 476 709 else 477 rc = ops->node_get(&fn, devmap_handle, index);710 rc = ops->node_get(&fn, service_id, index); 478 711 on_error(rc, goto out_with_answer); 479 712 … … 489 722 aoff64_t size = ops->size_get(fn); 490 723 async_answer_5(rid, fs_handle, 491 devmap_handle,724 service_id, 492 725 ops->index_get(fn), 493 726 LOWER32(size), … … 515 748 if (rc == EOK) { 516 749 aoff64_t size = ops->size_get(cur); 517 async_answer_5(rid, fs_handle, devmap_handle,750 async_answer_5(rid, fs_handle, service_id, 518 751 ops->index_get(cur), LOWER32(size), UPPER32(size), 519 752 old_lnkcnt); … … 553 786 if (rc == EOK) { 554 787 aoff64_t size = ops->size_get(cur); 555 async_answer_5(rid, fs_handle, devmap_handle,788 async_answer_5(rid, fs_handle, service_id, 556 789 ops->index_get(cur), LOWER32(size), UPPER32(size), 557 790 ops->lnkcnt_get(cur)); … … 577 810 ipc_call_t *request) 578 811 { 579 devmap_handle_t devmap_handle = (devmap_handle_t) IPC_GET_ARG1(*request);812 service_id_t service_id = (service_id_t) IPC_GET_ARG1(*request); 580 813 fs_index_t index = (fs_index_t) IPC_GET_ARG2(*request); 581 814 582 815 fs_node_t *fn; 583 int rc = ops->node_get(&fn, devmap_handle, index);816 int rc = ops->node_get(&fn, service_id, index); 584 817 on_error(rc, answer_and_return(rid, rc)); 585 818 … … 598 831 599 832 stat.fs_handle = fs_handle; 600 stat. devmap_handle = devmap_handle;833 stat.service_id = service_id; 601 834 stat.index = index; 602 835 stat.lnkcnt = ops->lnkcnt_get(fn); … … 604 837 stat.is_directory = ops->is_directory(fn); 605 838 stat.size = ops->size_get(fn); 606 stat. device = ops->device_get(fn);839 stat.service = ops->service_get(fn); 607 840 608 841 ops->node_put(fn); … … 623 856 ipc_call_t *request) 624 857 { 625 devmap_handle_t devmap_handle= IPC_GET_ARG1(*request);858 service_id_t service_id = IPC_GET_ARG1(*request); 626 859 fs_index_t index = IPC_GET_ARG2(*request); 627 860 628 861 fs_node_t *fn; 629 int rc = ops->node_get(&fn, devmap_handle, index);862 int rc = ops->node_get(&fn, service_id, index); 630 863 on_error(rc, answer_and_return(rid, rc)); 631 864 … … 637 870 rc = ops->node_open(fn); 638 871 aoff64_t size = ops->size_get(fn); 639 async_answer_4(rid, rc, LOWER32(size), UPPER32(size), ops->lnkcnt_get(fn), 872 async_answer_4(rid, rc, LOWER32(size), UPPER32(size), 873 ops->lnkcnt_get(fn), 640 874 (ops->is_file(fn) ? L_FILE : 0) | (ops->is_directory(fn) ? L_DIRECTORY : 0)); 641 875
Note:
See TracChangeset
for help on using the changeset viewer.