Linux-2.6.12-rc2
[linux-2.6/next.git] / arch / ia64 / sn / pci / pcibr / pcibr_ate.c
blob9d6854666f9b55c5716fe246e4bff3e2bc596d79
1 /*
2 * This file is subject to the terms and conditions of the GNU General Public
3 * License. See the file "COPYING" in the main directory of this archive
4 * for more details.
6 * Copyright (C) 2001-2004 Silicon Graphics, Inc. All rights reserved.
7 */
9 #include <linux/types.h>
10 #include <asm/sn/sn_sal.h>
11 #include "pci/pcibus_provider_defs.h"
12 #include "pci/pcidev.h"
13 #include "pci/pcibr_provider.h"
15 int pcibr_invalidate_ate = 0; /* by default don't invalidate ATE on free */
18 * mark_ate: Mark the ate as either free or inuse.
20 static void mark_ate(struct ate_resource *ate_resource, int start, int number,
21 uint64_t value)
24 uint64_t *ate = ate_resource->ate;
25 int index;
26 int length = 0;
28 for (index = start; length < number; index++, length++)
29 ate[index] = value;
34 * find_free_ate: Find the first free ate index starting from the given
35 * index for the desired consequtive count.
37 static int find_free_ate(struct ate_resource *ate_resource, int start,
38 int count)
41 uint64_t *ate = ate_resource->ate;
42 int index;
43 int start_free;
45 for (index = start; index < ate_resource->num_ate;) {
46 if (!ate[index]) {
47 int i;
48 int free;
49 free = 0;
50 start_free = index; /* Found start free ate */
51 for (i = start_free; i < ate_resource->num_ate; i++) {
52 if (!ate[i]) { /* This is free */
53 if (++free == count)
54 return start_free;
55 } else {
56 index = i + 1;
57 break;
60 } else
61 index++; /* Try next ate */
64 return -1;
68 * free_ate_resource: Free the requested number of ATEs.
70 static inline void free_ate_resource(struct ate_resource *ate_resource,
71 int start)
74 mark_ate(ate_resource, start, ate_resource->ate[start], 0);
75 if ((ate_resource->lowest_free_index > start) ||
76 (ate_resource->lowest_free_index < 0))
77 ate_resource->lowest_free_index = start;
82 * alloc_ate_resource: Allocate the requested number of ATEs.
84 static inline int alloc_ate_resource(struct ate_resource *ate_resource,
85 int ate_needed)
88 int start_index;
91 * Check for ate exhaustion.
93 if (ate_resource->lowest_free_index < 0)
94 return -1;
97 * Find the required number of free consequtive ates.
99 start_index =
100 find_free_ate(ate_resource, ate_resource->lowest_free_index,
101 ate_needed);
102 if (start_index >= 0)
103 mark_ate(ate_resource, start_index, ate_needed, ate_needed);
105 ate_resource->lowest_free_index =
106 find_free_ate(ate_resource, ate_resource->lowest_free_index, 1);
108 return start_index;
112 * Allocate "count" contiguous Bridge Address Translation Entries
113 * on the specified bridge to be used for PCI to XTALK mappings.
114 * Indices in rm map range from 1..num_entries. Indicies returned
115 * to caller range from 0..num_entries-1.
117 * Return the start index on success, -1 on failure.
119 int pcibr_ate_alloc(struct pcibus_info *pcibus_info, int count)
121 int status = 0;
122 uint64_t flag;
124 flag = pcibr_lock(pcibus_info);
125 status = alloc_ate_resource(&pcibus_info->pbi_int_ate_resource, count);
127 if (status < 0) {
128 /* Failed to allocate */
129 pcibr_unlock(pcibus_info, flag);
130 return -1;
133 pcibr_unlock(pcibus_info, flag);
135 return status;
139 * Setup an Address Translation Entry as specified. Use either the Bridge
140 * internal maps or the external map RAM, as appropriate.
142 static inline uint64_t *pcibr_ate_addr(struct pcibus_info *pcibus_info,
143 int ate_index)
145 if (ate_index < pcibus_info->pbi_int_ate_size) {
146 return pcireg_int_ate_addr(pcibus_info, ate_index);
148 panic("pcibr_ate_addr: invalid ate_index 0x%x", ate_index);
152 * Update the ate.
154 void inline
155 ate_write(struct pcibus_info *pcibus_info, int ate_index, int count,
156 volatile uint64_t ate)
158 while (count-- > 0) {
159 if (ate_index < pcibus_info->pbi_int_ate_size) {
160 pcireg_int_ate_set(pcibus_info, ate_index, ate);
161 } else {
162 panic("ate_write: invalid ate_index 0x%x", ate_index);
164 ate_index++;
165 ate += IOPGSIZE;
168 pcireg_tflush_get(pcibus_info); /* wait until Bridge PIO complete */
171 void pcibr_ate_free(struct pcibus_info *pcibus_info, int index)
174 volatile uint64_t ate;
175 int count;
176 uint64_t flags;
178 if (pcibr_invalidate_ate) {
179 /* For debugging purposes, clear the valid bit in the ATE */
180 ate = *pcibr_ate_addr(pcibus_info, index);
181 count = pcibus_info->pbi_int_ate_resource.ate[index];
182 ate_write(pcibus_info, index, count, (ate & ~PCI32_ATE_V));
185 flags = pcibr_lock(pcibus_info);
186 free_ate_resource(&pcibus_info->pbi_int_ate_resource, index);
187 pcibr_unlock(pcibus_info, flags);