iPXE
blocktrans.c
Go to the documentation of this file.
1 /*
2  * Copyright (C) 2015 Michael Brown <mbrown@fensystems.co.uk>.
3  *
4  * This program is free software; you can redistribute it and/or
5  * modify it under the terms of the GNU General Public License as
6  * published by the Free Software Foundation; either version 2 of the
7  * License, or any later version.
8  *
9  * This program is distributed in the hope that it will be useful, but
10  * WITHOUT ANY WARRANTY; without even the implied warranty of
11  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
12  * General Public License for more details.
13  *
14  * You should have received a copy of the GNU General Public License
15  * along with this program; if not, write to the Free Software
16  * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA
17  * 02110-1301, USA.
18  *
19  * You can also choose to distribute this program under the terms of
20  * the Unmodified Binary Distribution Licence (as given in the file
21  * COPYING.UBDL), provided that you have satisfied its requirements.
22  */
23 
24 FILE_LICENCE ( GPL2_OR_LATER_OR_UBDL );
25 
26 /**
27  * @file
28  *
29  * Block device translator
30  *
31  */
32 
33 #include <stdlib.h>
34 #include <string.h>
35 #include <errno.h>
36 #include <assert.h>
37 #include <ipxe/iobuf.h>
38 #include <ipxe/xfer.h>
39 #include <ipxe/blockdev.h>
40 #include <ipxe/blocktrans.h>
41 
42 /**
43  * Close block device translator
44  *
45  * @v blktrans Block device translator
46  * @v rc Reason for close
47  */
48 static void blktrans_close ( struct block_translator *blktrans, int rc ) {
49  struct block_device_capacity capacity;
50 
51  /* Report block device capacity, if applicable */
52  if ( ( rc == 0 ) && ( blktrans->blksize ) ) {
53 
54  /* Construct block device capacity */
55  capacity.blocks =
56  ( blktrans->xferbuf.len / blktrans->blksize );
57  capacity.blksize = blktrans->blksize;
58  capacity.max_count = -1U;
59 
60  /* Report block device capacity */
61  block_capacity ( &blktrans->block, &capacity );
62  }
63 
64  /* Shut down interfaces */
65  intf_shutdown ( &blktrans->xfer, rc );
66  intf_shutdown ( &blktrans->block, rc );
67 }
68 
69 /**
70  * Deliver data
71  *
72  * @v blktrans Block device translator
73  * @v iobuf I/O buffer
74  * @v meta Data transfer metadata
75  * @ret rc Return status code
76  */
77 static int blktrans_deliver ( struct block_translator *blktrans,
78  struct io_buffer *iobuf,
79  struct xfer_metadata *meta ) {
80  int rc;
81 
82  /* Deliver to buffer */
83  if ( ( rc = xferbuf_deliver ( &blktrans->xferbuf, iob_disown ( iobuf ),
84  meta ) ) != 0 ) {
85  DBGC ( blktrans, "BLKTRANS %p could not deliver: %s\n",
86  blktrans, strerror ( rc ) );
87  goto err;
88  }
89 
90  return 0;
91 
92  err:
93  blktrans_close ( blktrans, rc );
94  return rc;
95 }
96 
97 /**
98  * Get underlying data transfer buffer
99  *
100  * @v blktrans Block device translator
101  * @ret xferbuf Data transfer buffer
102  */
103 static struct xfer_buffer *
104 blktrans_buffer ( struct block_translator *blktrans ) {
105 
106  return &blktrans->xferbuf;
107 }
108 
109 /** Block device translator block device interface operations */
112 };
113 
114 /** Block device translator block device interface descriptor */
118 
119 /** Block device translator data transfer interface operations */
124 };
125 
126 /** Block device translator data transfer interface descriptor */
128  INTF_DESC_PASSTHRU ( struct block_translator, xfer,
130 
131 /**
132  * Insert block device translator
133  *
134  * @v block Block device interface
135  * @v buffer Data buffer (or NULL)
136  * @v size Length of data buffer, or block size
137  * @ret rc Return status code
138  */
139 int block_translate ( struct interface *block, void *buffer, size_t size ) {
140  struct block_translator *blktrans;
141  int rc;
142 
143  /* Allocate and initialise structure */
144  blktrans = zalloc ( sizeof ( *blktrans ) );
145  if ( ! blktrans ) {
146  rc = -ENOMEM;
147  goto err_alloc;
148  }
149  ref_init ( &blktrans->refcnt, NULL );
150  intf_init ( &blktrans->block, &blktrans_block_desc, &blktrans->refcnt );
151  intf_init ( &blktrans->xfer, &blktrans_xfer_desc, &blktrans->refcnt );
152  if ( buffer ) {
153  xferbuf_fixed_init ( &blktrans->xferbuf, buffer, size );
154  } else {
155  xferbuf_void_init ( &blktrans->xferbuf );
156  blktrans->blksize = size;
157  }
158 
159  /* Attach to interfaces, mortalise self, and return */
160  intf_insert ( block, &blktrans->block, &blktrans->xfer );
161  ref_put ( &blktrans->refcnt );
162 
163  DBGC2 ( blktrans, "BLKTRANS %p created", blktrans );
164  if ( buffer ) {
165  DBGC2 ( blktrans, " for %#lx+%#zx",
166  virt_to_phys ( buffer ), size );
167  }
168  DBGC2 ( blktrans, "\n" );
169  return 0;
170 
171  ref_put ( &blktrans->refcnt );
172  err_alloc:
173  return rc;
174 }
An object interface operation.
Definition: interface.h:17
struct refcnt refcnt
Reference count.
Definition: blocktrans.h:20
struct arbelprm_rc_send_wqe rc
Definition: arbel.h:14
void intf_close(struct interface *intf, int rc)
Close an object interface.
Definition: interface.c:249
int xferbuf_deliver(struct xfer_buffer *xferbuf, struct io_buffer *iobuf, struct xfer_metadata *meta)
Add received data to data transfer buffer.
Definition: xferbuf.c:174
Data transfer metadata.
Definition: xfer.h:22
void intf_shutdown(struct interface *intf, int rc)
Shut down an object interface.
Definition: interface.c:278
unsigned int max_count
Maximum number of blocks per single transfer.
Definition: blockdev.h:23
static struct interface_descriptor blktrans_block_desc
Block device translator block device interface descriptor.
Definition: blocktrans.c:115
static void blktrans_close(struct block_translator *blktrans, int rc)
Close block device translator.
Definition: blocktrans.c:48
#define ref_init(refcnt, free)
Initialise a reference counter.
Definition: refcnt.h:64
Error codes.
A data transfer buffer.
Definition: xferbuf.h:18
I/O buffers.
void block_capacity(struct interface *intf, struct block_device_capacity *capacity)
Report block device capacity.
Definition: blockdev.c:129
uint16_t size
Buffer size.
Definition: dwmac.h:14
#define DBGC(...)
Definition: compiler.h:505
Block device translator.
uint32_t buffer
Buffer index (or NETVSC_RNDIS_NO_BUFFER)
Definition: netvsc.h:16
static void xferbuf_void_init(struct xfer_buffer *xferbuf)
Initialise void data transfer buffer.
Definition: xferbuf.h:95
Data transfer interfaces.
size_t len
Size of data.
Definition: xferbuf.h:22
#define ENOMEM
Not enough space.
Definition: errno.h:534
#define iob_disown(iobuf)
Disown an I/O buffer.
Definition: iobuf.h:216
Assertions.
An object interface.
Definition: interface.h:124
A block device translator.
Definition: blocktrans.h:18
uint64_t blocks
Total number of blocks.
Definition: blockdev.h:19
static struct xfer_buffer * blktrans_buffer(struct block_translator *blktrans)
Get underlying data transfer buffer.
Definition: blocktrans.c:104
struct interface xfer
Data transfer interface.
Definition: blocktrans.h:24
An object interface descriptor.
Definition: interface.h:55
Block devices.
char * strerror(int errno)
Retrieve string representation of error number.
Definition: strerror.c:78
void * zalloc(size_t size)
Allocate cleared memory.
Definition: malloc.c:661
#define INTF_OP(op_type, object_type, op_func)
Define an object interface operation.
Definition: interface.h:32
int xfer_deliver(struct interface *intf, struct io_buffer *iobuf, struct xfer_metadata *meta)
Deliver datagram.
Definition: xfer.c:194
static struct interface_descriptor blktrans_xfer_desc
Block device translator data transfer interface descriptor.
Definition: blocktrans.c:127
static int blktrans_deliver(struct block_translator *blktrans, struct io_buffer *iobuf, struct xfer_metadata *meta)
Deliver data.
Definition: blocktrans.c:77
static struct interface_operation blktrans_xfer_operations[]
Block device translator data transfer interface operations.
Definition: blocktrans.c:120
int block_translate(struct interface *block, void *buffer, size_t size)
Insert block device translator.
Definition: blocktrans.c:139
static struct interface_operation blktrans_block_operations[]
Block device translator block device interface operations.
Definition: blocktrans.c:110
void intf_insert(struct interface *intf, struct interface *upper, struct interface *lower)
Insert a filter interface.
Definition: interface.c:401
Block device capacity.
Definition: blockdev.h:17
#define DBGC2(...)
Definition: compiler.h:522
uint8_t block[3][8]
DES-encrypted blocks.
Definition: mschapv2.h:12
struct xfer_buffer xferbuf
Data transfer buffer.
Definition: blocktrans.h:27
#define INTF_DESC_PASSTHRU(object_type, intf, operations, passthru)
Define an object interface descriptor with pass-through interface.
Definition: interface.h:97
size_t blksize
Block size.
Definition: blocktrans.h:29
uint8_t meta
Metadata flags.
Definition: ena.h:14
static void intf_init(struct interface *intf, struct interface_descriptor *desc, struct refcnt *refcnt)
Initialise an object interface.
Definition: interface.h:203
static void xferbuf_fixed_init(struct xfer_buffer *xferbuf, void *data, size_t len)
Initialise fixed-size data transfer buffer.
Definition: xferbuf.h:80
#define NULL
NULL pointer (VOID *)
Definition: Base.h:321
String functions.
#define ref_put(refcnt)
Drop reference to object.
Definition: refcnt.h:106
size_t blksize
Block size.
Definition: blockdev.h:21
FILE_LICENCE(GPL2_OR_LATER_OR_UBDL)
struct interface block
Block device interface.
Definition: blocktrans.h:22
A persistent I/O buffer.
Definition: iobuf.h:37