blob: 935029fc400d73a632a959b08962ad6b760fdf28 [file] [log] [blame]
Linus Torvalds1da177e2005-04-16 15:20:36 -07001/*
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.
5 *
Jes Sorensen8ed9b2c2006-02-13 05:29:57 -05006 * Copyright (C) 2001-2006 Silicon Graphics, Inc. All rights reserved.
Linus Torvalds1da177e2005-04-16 15:20:36 -07007 */
8
9#include <linux/types.h>
10#include <asm/sn/sn_sal.h>
Prarit Bhargavac13cf372005-07-06 15:26:51 -070011#include <asm/sn/pcibr_provider.h>
Mark Maule9b08ebd2005-04-25 11:32:16 -070012#include <asm/sn/pcibus_provider_defs.h>
13#include <asm/sn/pcidev.h>
Linus Torvalds1da177e2005-04-16 15:20:36 -070014
Jes Sorensen8ed9b2c2006-02-13 05:29:57 -050015int pcibr_invalidate_ate; /* by default don't invalidate ATE on free */
Linus Torvalds1da177e2005-04-16 15:20:36 -070016
17/*
18 * mark_ate: Mark the ate as either free or inuse.
19 */
20static void mark_ate(struct ate_resource *ate_resource, int start, int number,
Prarit Bhargava53493dc2006-01-16 19:54:40 -080021 u64 value)
Linus Torvalds1da177e2005-04-16 15:20:36 -070022{
Prarit Bhargava53493dc2006-01-16 19:54:40 -080023 u64 *ate = ate_resource->ate;
Linus Torvalds1da177e2005-04-16 15:20:36 -070024 int index;
25 int length = 0;
26
27 for (index = start; length < number; index++, length++)
28 ate[index] = value;
Linus Torvalds1da177e2005-04-16 15:20:36 -070029}
30
31/*
32 * find_free_ate: Find the first free ate index starting from the given
33 * index for the desired consequtive count.
34 */
35static int find_free_ate(struct ate_resource *ate_resource, int start,
36 int count)
37{
Prarit Bhargava53493dc2006-01-16 19:54:40 -080038 u64 *ate = ate_resource->ate;
Linus Torvalds1da177e2005-04-16 15:20:36 -070039 int index;
40 int start_free;
41
42 for (index = start; index < ate_resource->num_ate;) {
43 if (!ate[index]) {
44 int i;
45 int free;
46 free = 0;
47 start_free = index; /* Found start free ate */
48 for (i = start_free; i < ate_resource->num_ate; i++) {
49 if (!ate[i]) { /* This is free */
50 if (++free == count)
51 return start_free;
52 } else {
53 index = i + 1;
54 break;
55 }
56 }
57 } else
58 index++; /* Try next ate */
59 }
60
61 return -1;
62}
63
64/*
65 * free_ate_resource: Free the requested number of ATEs.
66 */
67static inline void free_ate_resource(struct ate_resource *ate_resource,
68 int start)
69{
Linus Torvalds1da177e2005-04-16 15:20:36 -070070 mark_ate(ate_resource, start, ate_resource->ate[start], 0);
71 if ((ate_resource->lowest_free_index > start) ||
72 (ate_resource->lowest_free_index < 0))
73 ate_resource->lowest_free_index = start;
Linus Torvalds1da177e2005-04-16 15:20:36 -070074}
75
76/*
77 * alloc_ate_resource: Allocate the requested number of ATEs.
78 */
79static inline int alloc_ate_resource(struct ate_resource *ate_resource,
80 int ate_needed)
81{
Linus Torvalds1da177e2005-04-16 15:20:36 -070082 int start_index;
83
84 /*
85 * Check for ate exhaustion.
86 */
87 if (ate_resource->lowest_free_index < 0)
88 return -1;
89
90 /*
91 * Find the required number of free consequtive ates.
92 */
93 start_index =
94 find_free_ate(ate_resource, ate_resource->lowest_free_index,
95 ate_needed);
96 if (start_index >= 0)
97 mark_ate(ate_resource, start_index, ate_needed, ate_needed);
98
99 ate_resource->lowest_free_index =
100 find_free_ate(ate_resource, ate_resource->lowest_free_index, 1);
101
102 return start_index;
103}
104
105/*
106 * Allocate "count" contiguous Bridge Address Translation Entries
107 * on the specified bridge to be used for PCI to XTALK mappings.
108 * Indices in rm map range from 1..num_entries. Indicies returned
109 * to caller range from 0..num_entries-1.
110 *
111 * Return the start index on success, -1 on failure.
112 */
113int pcibr_ate_alloc(struct pcibus_info *pcibus_info, int count)
114{
Jes Sorensen8ed9b2c2006-02-13 05:29:57 -0500115 int status;
116 unsigned long flags;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700117
Jes Sorensen8ed9b2c2006-02-13 05:29:57 -0500118 spin_lock_irqsave(&pcibus_info->pbi_lock, flags);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700119 status = alloc_ate_resource(&pcibus_info->pbi_int_ate_resource, count);
Jes Sorensen8ed9b2c2006-02-13 05:29:57 -0500120 spin_unlock_irqrestore(&pcibus_info->pbi_lock, flags);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700121
122 return status;
123}
124
125/*
126 * Setup an Address Translation Entry as specified. Use either the Bridge
127 * internal maps or the external map RAM, as appropriate.
128 */
Al Viroa9f627c2006-10-10 22:46:27 +0100129static inline u64 __iomem *pcibr_ate_addr(struct pcibus_info *pcibus_info,
Linus Torvalds1da177e2005-04-16 15:20:36 -0700130 int ate_index)
131{
132 if (ate_index < pcibus_info->pbi_int_ate_size) {
133 return pcireg_int_ate_addr(pcibus_info, ate_index);
134 }
135 panic("pcibr_ate_addr: invalid ate_index 0x%x", ate_index);
136}
137
138/*
139 * Update the ate.
140 */
141void inline
142ate_write(struct pcibus_info *pcibus_info, int ate_index, int count,
Prarit Bhargava53493dc2006-01-16 19:54:40 -0800143 volatile u64 ate)
Linus Torvalds1da177e2005-04-16 15:20:36 -0700144{
145 while (count-- > 0) {
146 if (ate_index < pcibus_info->pbi_int_ate_size) {
147 pcireg_int_ate_set(pcibus_info, ate_index, ate);
148 } else {
149 panic("ate_write: invalid ate_index 0x%x", ate_index);
150 }
151 ate_index++;
152 ate += IOPGSIZE;
153 }
154
155 pcireg_tflush_get(pcibus_info); /* wait until Bridge PIO complete */
156}
157
158void pcibr_ate_free(struct pcibus_info *pcibus_info, int index)
159{
160
Prarit Bhargava53493dc2006-01-16 19:54:40 -0800161 volatile u64 ate;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700162 int count;
Alexey Dobriyanc53421b2006-09-30 23:27:37 -0700163 unsigned long flags;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700164
165 if (pcibr_invalidate_ate) {
166 /* For debugging purposes, clear the valid bit in the ATE */
167 ate = *pcibr_ate_addr(pcibus_info, index);
168 count = pcibus_info->pbi_int_ate_resource.ate[index];
169 ate_write(pcibus_info, index, count, (ate & ~PCI32_ATE_V));
170 }
171
Jes Sorensen8ed9b2c2006-02-13 05:29:57 -0500172 spin_lock_irqsave(&pcibus_info->pbi_lock, flags);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700173 free_ate_resource(&pcibus_info->pbi_int_ate_resource, index);
Jes Sorensen8ed9b2c2006-02-13 05:29:57 -0500174 spin_unlock_irqrestore(&pcibus_info->pbi_lock, flags);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700175}