summaryrefslogtreecommitdiff
path: root/release/src/linux/linux/include/asm-s390/idals.h
blob: 46f9e0a9485e75692f1ff32eb868967c3dd809c6 (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
/* 
   * File...........: linux/include/asm-s390x/idals.h
   * Author(s)......: Holger Smolinski <Holger.Smolinski@de.ibm.com>
   * Bugreports.to..: <Linux390@de.ibm.com>
   * (C) IBM Corporation, IBM Deutschland Entwicklung GmbH, 2000a
   
   * History of changes
   * 07/24/00 new file
 */
#include <linux/config.h>
#include <asm/irq.h>

#define IDA_SIZE_LOG 11 /* 11 for 2k , 12 for 4k */
#define IDA_BLOCK_SIZE (1L<<IDA_SIZE_LOG)

static inline addr_t *
idal_alloc ( int nridaws )
{
	if ( nridaws > 33 )
		BUG();
	return kmalloc(nridaws * sizeof(addr_t), GFP_ATOMIC | GFP_DMA );
}

static inline void 
idal_free ( addr_t *idal )
{
	kfree (idal);
}

#if defined(CONFIG_ARCH_S390X)
extern unsigned long __create_idal(unsigned long address, int count);
#endif

/*
 * Function: set_normalized_cda
 * sets the address of the data in CCW
 * if necessary it allocates an IDAL and sets sthe appropriate flags
 */
static inline int
set_normalized_cda(ccw1_t * ccw, unsigned long address)
{
	int ret = 0;

#if defined(CONFIG_ARCH_S390X)
	if (((address + ccw->count) >> 31) != 0) {
		if (ccw->flags & CCW_FLAG_IDA)
			BUG();
		address = __create_idal(address, ccw->count);
		if (address)
			ccw->flags |= CCW_FLAG_IDA;
		else
			ret = -ENOMEM;
	}
#endif
	ccw->cda = (__u32) address;
	return ret;
}

/*
 * Function: clear_normalized_cda
 * releases any allocated IDAL related to the CCW
 */
static inline void
clear_normalized_cda ( ccw1_t * ccw ) 
{
#if defined(CONFIG_ARCH_S390X)
	if ( ccw -> flags & CCW_FLAG_IDA ) {
		idal_free ( (addr_t *)(unsigned long) (ccw -> cda ));
		ccw -> flags &= ~CCW_FLAG_IDA;
	}
#endif
	ccw -> cda = 0;
}