123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198 |
- /*
- * pNFS Objects layout driver high level definitions
- *
- * Copyright (C) 2007 Panasas Inc. [year of first publication]
- * All rights reserved.
- *
- * Benny Halevy <bhalevy@panasas.com>
- * Boaz Harrosh <bharrosh@panasas.com>
- *
- * This program is free software; you can redistribute it and/or modify
- * it under the terms of the GNU General Public License version 2
- * See the file COPYING included with this distribution for more details.
- *
- * Redistribution and use in source and binary forms, with or without
- * modification, are permitted provided that the following conditions
- * are met:
- *
- * 1. Redistributions of source code must retain the above copyright
- * notice, this list of conditions and the following disclaimer.
- * 2. Redistributions in binary form must reproduce the above copyright
- * notice, this list of conditions and the following disclaimer in the
- * documentation and/or other materials provided with the distribution.
- * 3. Neither the name of the Panasas company nor the names of its
- * contributors may be used to endorse or promote products derived
- * from this software without specific prior written permission.
- *
- * THIS SOFTWARE IS PROVIDED ``AS IS'' AND ANY EXPRESS OR IMPLIED
- * WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF
- * MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
- * DISCLAIMED. IN NO EVENT SHALL THE REGENTS OR CONTRIBUTORS BE LIABLE
- * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR
- * CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
- * SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR
- * BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF
- * LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING
- * NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS
- * SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
- */
- #include <scsi/osd_initiator.h>
- #include "objlayout.h"
- #define NFSDBG_FACILITY NFSDBG_PNFS_LD
- /*
- * Create a objlayout layout structure for the given inode and return it.
- */
- struct pnfs_layout_hdr *
- objlayout_alloc_layout_hdr(struct inode *inode, gfp_t gfp_flags)
- {
- struct objlayout *objlay;
- objlay = kzalloc(sizeof(struct objlayout), gfp_flags);
- dprintk("%s: Return %p\n", __func__, objlay);
- return &objlay->pnfs_layout;
- }
- /*
- * Free an objlayout layout structure
- */
- void
- objlayout_free_layout_hdr(struct pnfs_layout_hdr *lo)
- {
- struct objlayout *objlay = OBJLAYOUT(lo);
- dprintk("%s: objlay %p\n", __func__, objlay);
- kfree(objlay);
- }
- /*
- * Unmarshall layout and store it in pnfslay.
- */
- struct pnfs_layout_segment *
- objlayout_alloc_lseg(struct pnfs_layout_hdr *pnfslay,
- struct nfs4_layoutget_res *lgr,
- gfp_t gfp_flags)
- {
- int status = -ENOMEM;
- struct xdr_stream stream;
- struct xdr_buf buf = {
- .pages = lgr->layoutp->pages,
- .page_len = lgr->layoutp->len,
- .buflen = lgr->layoutp->len,
- .len = lgr->layoutp->len,
- };
- struct page *scratch;
- struct pnfs_layout_segment *lseg;
- dprintk("%s: Begin pnfslay %p\n", __func__, pnfslay);
- scratch = alloc_page(gfp_flags);
- if (!scratch)
- goto err_nofree;
- xdr_init_decode(&stream, &buf, NULL);
- xdr_set_scratch_buffer(&stream, page_address(scratch), PAGE_SIZE);
- status = objio_alloc_lseg(&lseg, pnfslay, &lgr->range, &stream, gfp_flags);
- if (unlikely(status)) {
- dprintk("%s: objio_alloc_lseg Return err %d\n", __func__,
- status);
- goto err;
- }
- __free_page(scratch);
- dprintk("%s: Return %p\n", __func__, lseg);
- return lseg;
- err:
- __free_page(scratch);
- err_nofree:
- dprintk("%s: Err Return=>%d\n", __func__, status);
- return ERR_PTR(status);
- }
- /*
- * Free a layout segement
- */
- void
- objlayout_free_lseg(struct pnfs_layout_segment *lseg)
- {
- dprintk("%s: freeing layout segment %p\n", __func__, lseg);
- if (unlikely(!lseg))
- return;
- objio_free_lseg(lseg);
- }
- /*
- * Get Device Info API for io engines
- */
- struct objlayout_deviceinfo {
- struct page *page;
- struct pnfs_osd_deviceaddr da; /* This must be last */
- };
- /* Initialize and call nfs_getdeviceinfo, then decode and return a
- * "struct pnfs_osd_deviceaddr *" Eventually objlayout_put_deviceinfo()
- * should be called.
- */
- int objlayout_get_deviceinfo(struct pnfs_layout_hdr *pnfslay,
- struct nfs4_deviceid *d_id, struct pnfs_osd_deviceaddr **deviceaddr,
- gfp_t gfp_flags)
- {
- struct objlayout_deviceinfo *odi;
- struct pnfs_device pd;
- struct super_block *sb;
- struct page *page, **pages;
- u32 *p;
- int err;
- page = alloc_page(gfp_flags);
- if (!page)
- return -ENOMEM;
- pages = &page;
- pd.pages = pages;
- memcpy(&pd.dev_id, d_id, sizeof(*d_id));
- pd.layout_type = LAYOUT_OSD2_OBJECTS;
- pd.pages = &page;
- pd.pgbase = 0;
- pd.pglen = PAGE_SIZE;
- pd.mincount = 0;
- sb = pnfslay->plh_inode->i_sb;
- err = nfs4_proc_getdeviceinfo(NFS_SERVER(pnfslay->plh_inode), &pd);
- dprintk("%s nfs_getdeviceinfo returned %d\n", __func__, err);
- if (err)
- goto err_out;
- p = page_address(page);
- odi = kzalloc(sizeof(*odi), gfp_flags);
- if (!odi) {
- err = -ENOMEM;
- goto err_out;
- }
- pnfs_osd_xdr_decode_deviceaddr(&odi->da, p);
- odi->page = page;
- *deviceaddr = &odi->da;
- return 0;
- err_out:
- __free_page(page);
- return err;
- }
- void objlayout_put_deviceinfo(struct pnfs_osd_deviceaddr *deviceaddr)
- {
- struct objlayout_deviceinfo *odi = container_of(deviceaddr,
- struct objlayout_deviceinfo,
- da);
- __free_page(odi->page);
- kfree(odi);
- }
|